com.nimbusds.jose.jca.JCASupport Maven / Gradle / Ivy
/*
* nimbus-jose-jwt
*
* Copyright 2012-2016, Connect2id Ltd.
*
* 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.jca;
import java.security.NoSuchAlgorithmException;
import java.security.Provider;
import java.security.Security;
import javax.crypto.Cipher;
import javax.crypto.Mac;
import javax.crypto.NoSuchPaddingException;
import com.nimbusds.jose.*;
import com.nimbusds.jose.crypto.impl.ECDSA;
import com.nimbusds.jose.crypto.impl.RSASSA;
/**
* Java Cryptography Architecture (JCA) support helper.
*
* @version 2022-05-16
*/
public final class JCASupport {
/**
* Checks if unlimited cryptographic strength is supported. If not
* download the appropriate jurisdiction policy files for your Java
* edition:
*
* JCE Unlimited Strength Jurisdiction Policy Files for Java 7
*
*
JCE Unlimited Strength Jurisdiction Policy Files for Java 8
*
* @return {@code true} if unlimited cryptographic strength is
* supported, {@code false} if not.
*/
public static boolean isUnlimitedStrength() {
try {
return Cipher.getMaxAllowedKeyLength("AES") >= 256;
} catch (NoSuchAlgorithmException e) {
return false;
}
}
/**
* Checks if the specified JOSE algorithm is supported by the default
* system JCA provider(s).
*
* @param alg The JOSE algorithm. Must not be {@code null}.
*
* @return {@code true} if the JOSE algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final Algorithm alg) {
if (alg instanceof JWSAlgorithm) {
return isSupported((JWSAlgorithm)alg);
}
if (alg instanceof JWEAlgorithm) {
return isSupported((JWEAlgorithm)alg);
}
if (alg instanceof EncryptionMethod) {
return isSupported((EncryptionMethod)alg);
}
throw new IllegalArgumentException("Unexpected algorithm class: " + alg.getClass().getCanonicalName());
}
/**
* Checks if a JOSE algorithm is supported by the specified JCA
* provider.
*
* @param alg The JOSE algorithm. Must not be {@code null}.
* @param provider The JCA provider. Must not be {@code null}.
*
* @return {@code true} if the JOSE algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final Algorithm alg, final Provider provider) {
if (alg instanceof JWSAlgorithm) {
return isSupported((JWSAlgorithm)alg, provider);
}
if (alg instanceof JWEAlgorithm) {
return isSupported((JWEAlgorithm)alg, provider);
}
if (alg instanceof EncryptionMethod) {
return isSupported((EncryptionMethod)alg, provider);
}
throw new IllegalArgumentException("Unexpected algorithm class: " + alg.getClass().getCanonicalName());
}
/**
* Checks if the specified JWS algorithm is supported by the default
* system JCA provider(s).
*
* @param alg The JWS algorithm. Must not be {@code null}.
*
* @return {@code true} if the JWS algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final JWSAlgorithm alg) {
if (alg.getName().equals(Algorithm.NONE.getName())) {
return true;
}
for (Provider p: Security.getProviders()) {
if (isSupported(alg, p)) {
return true;
}
}
return false;
}
/**
* Checks if a JWS algorithm is supported by the specified JCA
* provider.
*
* @param alg The JWS algorithm. Must not be {@code null}.
* @param provider The JCA provider. Must not be {@code null}.
*
* @return {@code true} if the JWS algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final JWSAlgorithm alg, final Provider provider) {
if (JWSAlgorithm.Family.HMAC_SHA.contains(alg)) {
String jcaName;
if (alg.equals(JWSAlgorithm.HS256)) {
jcaName = "HMACSHA256";
} else if (alg.equals(JWSAlgorithm.HS384)) {
jcaName = "HMACSHA384";
} else if (alg.equals(JWSAlgorithm.HS512)) {
jcaName = "HMACSHA512";
} else {
return false;
}
try {
Mac.getInstance(jcaName, provider);
} catch (NoSuchAlgorithmException e) {
return false;
}
return true;
}
if (JWSAlgorithm.Family.RSA.contains(alg)) {
try {
RSASSA.getSignerAndVerifier(alg, provider);
} catch (JOSEException e) {
return false;
}
return true;
}
if (JWSAlgorithm.Family.EC.contains(alg)) {
try {
ECDSA.getSignerAndVerifier(alg, provider);
} catch (JOSEException e) {
return false;
}
return true;
}
return false;
}
/**
* Checks if the specified JWE algorithm is supported by the default
* system JCA provider(s).
*
* @param alg The JWE algorithm. Must not be {@code null}.
*
* @return {@code true} if the JWE algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final JWEAlgorithm alg) {
for (Provider p: Security.getProviders()) {
if (isSupported(alg, p)) {
return true;
}
}
return false;
}
/**
* Checks if a JWE algorithm is supported by the specified JCA
* provider.
*
* @param alg The JWE algorithm. Must not be {@code null}.
* @param provider The JCA provider. Must not be {@code null}.
*
* @return {@code true} if the JWE algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final JWEAlgorithm alg, final Provider provider) {
String jcaName;
if (JWEAlgorithm.Family.RSA.contains(alg)) {
if (alg.equals(JWEAlgorithm.RSA1_5)) {
jcaName = "RSA/ECB/PKCS1Padding";
} else if (alg.equals(JWEAlgorithm.RSA_OAEP)) {
jcaName = "RSA/ECB/OAEPWithSHA-1AndMGF1Padding";
} else if (alg.equals(JWEAlgorithm.RSA_OAEP_256)) {
jcaName = "RSA/ECB/OAEPWithSHA-256AndMGF1Padding";
} else if (alg.equals(JWEAlgorithm.RSA_OAEP_512)) {
jcaName = "RSA/ECB/OAEPWithSHA-512AndMGF1Padding";
} else {
return false;
}
// Do direct test
try {
Cipher.getInstance(jcaName, provider);
} catch (NoSuchAlgorithmException | NoSuchPaddingException e) {
return false;
}
return true;
}
if (JWEAlgorithm.Family.AES_KW.contains(alg)) {
return provider.getService("Cipher", "AESWrap") != null;
}
if (JWEAlgorithm.Family.ECDH_ES.contains(alg)) {
return provider.getService("KeyAgreement", "ECDH") != null;
}
if (JWEAlgorithm.Family.AES_GCM_KW.contains(alg)) {
// Do direct test
try {
Cipher.getInstance("AES/GCM/NoPadding", provider);
} catch (NoSuchAlgorithmException | NoSuchPaddingException e) {
return false;
}
return true;
}
if (JWEAlgorithm.Family.PBES2.contains(alg)) {
String hmac;
if (alg.equals(JWEAlgorithm.PBES2_HS256_A128KW)) {
hmac = "HmacSHA256";
} else if (alg.equals(JWEAlgorithm.PBES2_HS384_A192KW)) {
hmac = "HmacSHA384";
} else {
hmac = "HmacSHA512";
}
return provider.getService("KeyGenerator", hmac) != null;
}
return JWEAlgorithm.DIR.equals(alg); // Always supported
}
/**
* Checks if the specified JWE encryption method is supported by the
* default system JCA provider(s).
*
* @param enc The JWE encryption method. Must not be {@code null}.
*
* @return {@code true} if the JWE algorithm is supported, else
* {@code false}.
*/
public static boolean isSupported(final EncryptionMethod enc) {
for (Provider p: Security.getProviders()) {
if (isSupported(enc, p)) {
return true;
}
}
return false;
}
/**
* Checks if a JWE encryption method is supported by the specified
* JCA provider.
*
* @param enc The JWE encryption method. Must not be {@code null}.
* @param provider The JCA provider. Must not be {@code null}.
*
* @return {@code true} if the JWE encryption method is supported, else
* {@code false}.
*/
public static boolean isSupported(final EncryptionMethod enc, final Provider provider) {
if (EncryptionMethod.Family.AES_CBC_HMAC_SHA.contains(enc)) {
// Do direct test
try {
Cipher.getInstance("AES/CBC/PKCS5Padding", provider);
} catch (NoSuchAlgorithmException | NoSuchPaddingException e) {
return false;
}
// Check hmac
String hmac;
if (enc.equals(EncryptionMethod.A128CBC_HS256)) {
hmac = "HmacSHA256";
} else if (enc.equals(EncryptionMethod.A192CBC_HS384)) {
hmac = "HmacSHA384";
} else {
hmac = "HmacSHA512";
}
return provider.getService("KeyGenerator", hmac) != null;
}
if (EncryptionMethod.Family.AES_GCM.contains(enc)) {
// Do direct test
try {
Cipher.getInstance("AES/GCM/NoPadding", provider);
} catch (NoSuchAlgorithmException | NoSuchPaddingException e) {
return false;
}
return true;
}
return false;
}
/**
* Prevents public instantiation.
*/
private JCASupport() {
}
}