com.nimbusds.jose.crypto.impl.ECDH1PU Maven / Gradle / Ivy
/*
* nimbus-jose-jwt
*
* Copyright 2012-2021, Connect2id Ltd and contributors.
*
* 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.nimbusds.jose.crypto.impl;
import com.nimbusds.jose.*;
import com.nimbusds.jose.crypto.utils.ECChecks;
import com.nimbusds.jose.jwk.Curve;
import com.nimbusds.jose.jwk.OctetKeyPair;
import com.nimbusds.jose.util.Base64URL;
import com.nimbusds.jose.util.ByteUtils;
import javax.crypto.SecretKey;
import javax.crypto.spec.SecretKeySpec;
import java.nio.charset.StandardCharsets;
import java.security.Provider;
import java.security.interfaces.ECPrivateKey;
import java.security.interfaces.ECPublicKey;
import java.util.Objects;
/**
* Elliptic Curve Diffie-Hellman One-Pass Unified Model (ECDH-1PU) key
* agreement functions and utilities.
*
* @see Public
* Key Authenticated Encryption for JOSE: ECDH-1PU
*
* @author Alexander Martynov
* @version 2021-08-03
*/
public class ECDH1PU {
/**
* Resolves the ECDH algorithm mode.
*
* @param alg The JWE algorithm. Must be supported and not {@code null}.
*
* @return The algorithm mode.
*
* @throws JOSEException If the JWE algorithm is not supported.
*/
public static ECDH.AlgorithmMode resolveAlgorithmMode(final JWEAlgorithm alg)
throws JOSEException {
Objects.requireNonNull(alg, "The parameter \"alg\" must not be null");
if (alg.equals(JWEAlgorithm.ECDH_1PU)) {
return ECDH.AlgorithmMode.DIRECT;
}
if (alg.equals(JWEAlgorithm.ECDH_1PU_A128KW) ||
alg.equals(JWEAlgorithm.ECDH_1PU_A192KW) ||
alg.equals(JWEAlgorithm.ECDH_1PU_A256KW)
) {
return ECDH.AlgorithmMode.KW;
}
throw new JOSEException(AlgorithmSupportMessage.unsupportedJWEAlgorithm(
alg,
ECDHCryptoProvider.SUPPORTED_ALGORITHMS));
}
/**
* Returns the bit length of the shared key (derived via concat KDF)
* for the specified JWE ECDH algorithm.
*
* @param alg The JWE ECDH algorithm. Must be supported and not
* {@code null}.
* @param enc The encryption method. Must be supported and not
* {@code null}.
*
* @return The bit length of the shared key.
*
* @throws JOSEException If the JWE algorithm or encryption method is
* not supported.
*/
public static int sharedKeyLength(final JWEAlgorithm alg, final EncryptionMethod enc)
throws JOSEException {
Objects.requireNonNull(alg, "The parameter \"alg\" must not be null");
Objects.requireNonNull(enc, "The parameter \"enc\" must not be null");
if (alg.equals(JWEAlgorithm.ECDH_1PU)) {
int length = enc.cekBitLength();
if (length == 0) {
throw new JOSEException("Unsupported JWE encryption method " + enc);
}
return length;
}
if (alg.equals(JWEAlgorithm.ECDH_1PU_A128KW)) {
return 128;
}
if (alg.equals(JWEAlgorithm.ECDH_1PU_A192KW)) {
return 192;
}
if (alg.equals(JWEAlgorithm.ECDH_1PU_A256KW)) {
return 256;
}
throw new JOSEException(AlgorithmSupportMessage.unsupportedJWEAlgorithm(
alg, ECDHCryptoProvider.SUPPORTED_ALGORITHMS));
}
/**
* Derives a shared key (via concat KDF).
*
* The method should only be called in the
* {@link ECDH.AlgorithmMode#DIRECT} mode.
*
* The method derives the Content Encryption Key (CEK) for the "enc"
* algorithm, in the {@link ECDH.AlgorithmMode#DIRECT} mode.
*
* The method does not take the auth tag because the auth tag will be
* generated using a CEK derived as an output of this method.
*
* @param header The JWE header. Its algorithm and encryption method
* must be supported. Must not be {@code null}.
* @param Z The derived shared secret ('Z'). Must not be
* {@code null}.
* @param concatKDF The concat KDF. Must be initialised and not
* {@code null}.
*
* @return The derived shared key.
*
* @throws JOSEException If derivation of the shared key failed.
*/
public static SecretKey deriveSharedKey(final JWEHeader header,
final SecretKey Z,
final ConcatKDF concatKDF)
throws JOSEException {
Objects.requireNonNull(header, "The parameter \"header\" must not be null");
Objects.requireNonNull(Z, "The parameter \"Z\" must not be null");
Objects.requireNonNull(concatKDF, "The parameter \"concatKDF\" must not be null");
final int sharedKeyLength = sharedKeyLength(header.getAlgorithm(), header.getEncryptionMethod());
// Set the alg ID for the concat KDF
ECDH.AlgorithmMode algMode = resolveAlgorithmMode(header.getAlgorithm());
final String algID;
if (algMode == ECDH.AlgorithmMode.DIRECT) {
// algID = enc
algID = header.getEncryptionMethod().getName();
} else if (algMode == ECDH.AlgorithmMode.KW) {
// algID = alg
algID = header.getAlgorithm().getName();
} else {
throw new JOSEException("Unsupported JWE ECDH algorithm mode: " + algMode);
}
return concatKDF.deriveKey(
Z,
sharedKeyLength,
ConcatKDF.encodeDataWithLength(algID.getBytes(StandardCharsets.US_ASCII)),
ConcatKDF.encodeDataWithLength(header.getAgreementPartyUInfo()),
ConcatKDF.encodeDataWithLength(header.getAgreementPartyVInfo()),
ConcatKDF.encodeIntData(sharedKeyLength),
ConcatKDF.encodeNoData()
);
}
/**
* Derives a shared key (via concat KDF).
*
* The method should only be called in {@link ECDH.AlgorithmMode#KW}.
*
* In Key Agreement with {@link ECDH.AlgorithmMode#KW} mode, the JWE
* Authentication Tag is included in the input to the KDF. This ensures
* that the content of the JWE was produced by the original sender and not
* by another recipient.
*
*
* @param header The JWE header. Its algorithm and encryption method
* must be supported. Must not be {@code null}.
* @param Z The derived shared secret ('Z'). Must not be
* {@code null}.
* @param tag In Direct Key Agreement mode this is set to an empty
* octet string. In Key Agreement with Key Wrapping mode,
* this is set to a value of the form Data, where Data is
* the raw octets of the JWE Authentication Tag.
* @param concatKDF The concat KDF. Must be initialised and not
* {@code null}.
*
* @return The derived shared key.
*
* @throws JOSEException If derivation of the shared key failed.
*/
public static SecretKey deriveSharedKey(final JWEHeader header,
final SecretKey Z,
final Base64URL tag,
final ConcatKDF concatKDF)
throws JOSEException {
Objects.requireNonNull(header, "The parameter \"header\" must not be null");
Objects.requireNonNull(Z, "The parameter \"Z\" must not be null");
Objects.requireNonNull(tag, "The parameter \"tag\" must not be null");
Objects.requireNonNull(concatKDF, "The parameter \"concatKDF\" must not be null");
final int sharedKeyLength = sharedKeyLength(header.getAlgorithm(), header.getEncryptionMethod());
// Set the alg ID for the concat KDF
ECDH.AlgorithmMode algMode = resolveAlgorithmMode(header.getAlgorithm());
final String algID;
if (algMode == ECDH.AlgorithmMode.DIRECT) {
// algID = enc
algID = header.getEncryptionMethod().getName();
} else if (algMode == ECDH.AlgorithmMode.KW) {
// algID = alg
algID = header.getAlgorithm().getName();
} else {
throw new JOSEException("Unsupported JWE ECDH algorithm mode: " + algMode);
}
return concatKDF.deriveKey(
Z,
sharedKeyLength,
ConcatKDF.encodeDataWithLength(algID.getBytes(StandardCharsets.US_ASCII)),
ConcatKDF.encodeDataWithLength(header.getAgreementPartyUInfo()),
ConcatKDF.encodeDataWithLength(header.getAgreementPartyVInfo()),
ConcatKDF.encodeIntData(sharedKeyLength),
ConcatKDF.encodeNoData(),
ConcatKDF.encodeDataWithLength(tag)
);
}
/**
* Derives a shared secret (also called 'Z') where Z is the concatenation
* of Ze and Zs.
*
* @param Ze The shared secret derived from applying the ECDH primitive to
* the sender's ephemeral private key and the recipient's static
* public key (when sending) or the recipient's static private
* key and the sender's ephemeral public key (when receiving).
* Must not be {@code null}.
* @param Zs The shared secret derived from applying the ECDH primitive to
* the sender's static private key and the recipient's static
* public key (when sending) or the recipient's static private
* key and the sender's static public key (when receiving). Must
* not be {@code null}.
*
* @return The derived shared key.
*/
public static SecretKey deriveZ(final SecretKey Ze, final SecretKey Zs) {
Objects.requireNonNull(Ze, "The parameter \"Ze\" must not be null");
Objects.requireNonNull(Zs, "The parameter \"Zs\" must not be null");
byte[] encodedKey = ByteUtils.concat(Ze.getEncoded(), Zs.getEncoded());
return new SecretKeySpec(encodedKey, 0, encodedKey.length, "AES");
}
/**
* Derives a shared secret (also called 'Z') for sender where Z is the
* concatenation of Ze and Zs. Where Ze is shared secret from applying
* the ECDH primitive to the sender's ephemeral private key and the recipient's
* static public key, Zs is the shared secret derived from
* applying the ECDH primitive to the sender's static private key and
* the recipient's static public key.
*
* @param privateKey The sender EC private key.
* @param publicKey The recipient EC public key.
* @param epk The sender EC ephemeral private key.
* @param provider The JCA provider for the ECDH key agreement,
* {@code null} to use the default.
*
* @return The derived shared secret ('Z'), with algorithm "AES".
*
* @throws JOSEException If derivation of the shared secret failed.
*/
public static SecretKey deriveSenderZ(
final ECPrivateKey privateKey,
final ECPublicKey publicKey,
final ECPrivateKey epk,
final Provider provider) throws JOSEException {
validateSameCurve(privateKey, publicKey);
validateSameCurve(epk, publicKey);
SecretKey Ze = ECDH.deriveSharedSecret(
publicKey,
epk,
provider
);
SecretKey Zs = ECDH.deriveSharedSecret(
publicKey,
privateKey,
provider
);
return deriveZ(Ze, Zs);
}
/**
* Derives a shared secret (also called 'Z') for sender where Z is the
* concatenation of Ze and Zs. Where Ze is shared secret from applying
* the ECDH primitive to the sender's ephemeral public key and the recipient's
* static private key, Zs is the shared secret derived from
* applying the ECDH primitive to the sender's static public key and
* the recipient's static private key.
*
* @param privateKey The sender OctetKey private key.
* @param publicKey The recipient OctetKey public key.
* @param epk The sender OctetKey ephemeral private key.
*
* @return The derived shared secret ('Z'), with algorithm "AES".
*
* @throws JOSEException If derivation of the shared secret failed.
*/
public static SecretKey deriveSenderZ(
final OctetKeyPair privateKey,
final OctetKeyPair publicKey,
final OctetKeyPair epk) throws JOSEException {
validateSameCurve(privateKey, publicKey);
validateSameCurve(epk, publicKey);
SecretKey Ze = ECDH.deriveSharedSecret(publicKey, epk);
SecretKey Zs = ECDH.deriveSharedSecret(publicKey, privateKey);
return deriveZ(Ze, Zs);
}
/**
* Derives a shared secret (also called 'Z') for sender where Z is the
* concatenation of Ze and Zs. Where Ze is shared secret from applying
* the ECDH primitive to the sender's ephemeral public key and the recipient's
* static private key, Zs is the shared secret derived from
* applying the ECDH primitive to the sender's static public key and
* the recipient's static private key.
*
* @param privateKey The sender EC private key.
* @param publicKey The recipient EC public key.
* @param epk The sender EC ephemeral public key.
* @param provider The JCA provider for the ECDH key agreement,
* {@code null} to use the default.
*
* @return The derived shared secret ('Z'), with algorithm "AES".
*
* @throws JOSEException If derivation of the shared secret failed.
*/
public static SecretKey deriveRecipientZ(
final ECPrivateKey privateKey,
final ECPublicKey publicKey,
final ECPublicKey epk,
final Provider provider) throws JOSEException {
validateSameCurve(privateKey, publicKey);
validateSameCurve(privateKey, epk);
SecretKey Ze = ECDH.deriveSharedSecret(
epk,
privateKey,
provider
);
SecretKey Zs = ECDH.deriveSharedSecret(
publicKey,
privateKey,
provider
);
return deriveZ(Ze, Zs);
}
/**
* Derives a shared secret (also called 'Z') for recipient where Z is the
* concatenation of Ze and Zs.
*
* @param privateKey The sender OctetKey private key.
* @param publicKey The recipient OctetKey public key.
* @param epk The sender OctetKey ephemeral private key.
*
* @return The derived shared secret ('Z'), with algorithm "AES".
*
* @throws JOSEException If derivation of the shared secret failed.
*/
public static SecretKey deriveRecipientZ(
final OctetKeyPair privateKey,
final OctetKeyPair publicKey,
final OctetKeyPair epk) throws JOSEException {
validateSameCurve(privateKey, publicKey);
validateSameCurve(privateKey, epk);
SecretKey Ze = ECDH.deriveSharedSecret(
epk,
privateKey
);
SecretKey Zs = ECDH.deriveSharedSecret(
publicKey,
privateKey
);
return deriveZ(Ze, Zs);
}
/**
* Ensures the private key and public key are from the same curve.
*
* @param privateKey EC private key. Must not be {@code null}.
* @param publicKey EC public key. Must not be {@code null}.
*
* @throws JOSEException If the key curves don't match.
*/
public static void validateSameCurve(final ECPrivateKey privateKey, final ECPublicKey publicKey)
throws JOSEException{
Objects.requireNonNull(privateKey, "The parameter \"privateKey\" must not be null");
Objects.requireNonNull(publicKey, "The parameter \"publicKey\" must not be null");
if (!privateKey.getParams().getCurve().equals(publicKey.getParams().getCurve())) {
throw new JOSEException("Curve of public key does not match curve of private key");
}
if (!ECChecks.isPointOnCurve(publicKey, privateKey)) {
throw new JOSEException("Invalid public EC key: Point(s) not on the expected curve");
}
}
/**
* Ensures the private key and public key are from the same curve.
*
* @param privateKey OKP private key. Must not be {@code null}.
* @param publicKey OKP public key. Must not be {@code null}.
*
* @throws JOSEException If the curves don't match.
*/
public static void validateSameCurve(final OctetKeyPair privateKey, final OctetKeyPair publicKey)
throws JOSEException {
Objects.requireNonNull(privateKey, "The parameter \"privateKey\" must not be null");
Objects.requireNonNull(publicKey, "The parameter \"publicKey\" must not be null");
if (!privateKey.isPrivate()) {
throw new JOSEException("OKP private key should be a private key");
}
if (publicKey.isPrivate()) {
throw new JOSEException("OKP public key should not be a private key");
}
if (!publicKey.getCurve().equals(Curve.X25519)) {
throw new JOSEException("Only supports OctetKeyPairs with crv=X25519");
}
if (!privateKey.getCurve().equals(publicKey.getCurve())) {
throw new JOSEException("Curve of public key does not match curve of private key");
}
}
/**
* Prevents public instantiation.
*/
private ECDH1PU() {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy