Skip to content

Commit

Permalink
fix: convert RSA private key to JWK (#4055)
Browse files Browse the repository at this point in the history
fix: allow conversion RSA->JWK if only private key is given
  • Loading branch information
paullatzelsperger authored Mar 27, 2024
1 parent f2ca192 commit e372688
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 2 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@
import java.security.interfaces.EdECKey;
import java.security.interfaces.EdECPrivateKey;
import java.security.interfaces.EdECPublicKey;
import java.security.interfaces.RSAPrivateCrtKey;
import java.security.interfaces.RSAPublicKey;
import java.security.spec.ECGenParameterSpec;
import java.security.spec.ECPoint;
Expand Down Expand Up @@ -302,8 +303,24 @@ private static Curve getCurveAllowing(EdECKey edKey, String... allowedCurves) {
}

private static RSAKey convertRsaKey(KeyPair keypair, @Nullable String kid) {
return new RSAKey.Builder((RSAPublicKey) keypair.getPublic())
.privateKey(keypair.getPrivate())

if (keypair.getPublic() == null && keypair.getPrivate() == null) {
throw new IllegalArgumentException("Either the public or the private key of a keypair must be non-null when converting RSA -> JWK");
}
var key = Optional.ofNullable(keypair.getPublic()).orElseGet(() -> {
var keySpec = new java.security.spec.RSAPublicKeySpec(((RSAPrivateCrtKey) keypair.getPrivate()).getModulus(), ((RSAPrivateCrtKey) keypair.getPrivate()).getPublicExponent());
try {
var gen = KeyFactory.getInstance("RSA");
return gen.generatePublic(keySpec);
} catch (NoSuchAlgorithmException | InvalidKeySpecException e) {
throw new RuntimeException(e);
}
});
var builder = new RSAKey.Builder((RSAPublicKey) key);
if (keypair.getPrivate() != null) {
builder.privateKey(keypair.getPrivate());
}
return builder
.keyID(kid)
.keyUse(KeyUse.SIGNATURE).build();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,33 @@ void createJwk_rsaKey() throws NoSuchAlgorithmException {
assertThat(jwk.getKeyID()).isNull();
}

@Test
void createJwk_rsaKey_onlyPrivate() throws NoSuchAlgorithmException {
var pk = createRsa();
var keypair = new KeyPair(null, pk.getPrivate());
var jwk = CryptoConverter.createJwk(keypair);
assertThat(jwk).isInstanceOf(RSAKey.class);
assertThat(jwk.isPrivate()).isTrue();
assertThat(jwk.getKeyID()).isNull();
}

@Test
void createJwk_rsaKey_whenEmptyKeypair() throws NoSuchAlgorithmException {
var keypair = new KeyPair(null, null);
assertThatThrownBy(() -> CryptoConverter.createJwk(keypair))
.isInstanceOf(IllegalArgumentException.class);
}

@Test
void createJwk_rsaKey_onlyPublic() throws NoSuchAlgorithmException {
var pk = createRsa();
var keypair = new KeyPair(pk.getPublic(), null);
var jwk = CryptoConverter.createJwk(keypair);
assertThat(jwk).isInstanceOf(RSAKey.class);
assertThat(jwk.isPrivate()).isFalse();
assertThat(jwk.getKeyID()).isNull();
}

@Test
void createJwk_rsaKey_withKeyId() throws NoSuchAlgorithmException {
var pk = createRsa();
Expand Down

0 comments on commit e372688

Please sign in to comment.