KEYCLOAK-12747 JWKTest.publicEs256 sometimes failing
This commit is contained in:
parent
30e735dd25
commit
f3e2b6d040
3 changed files with 70 additions and 35 deletions
|
@ -26,13 +26,14 @@ import org.keycloak.crypto.Algorithm;
|
||||||
import org.keycloak.crypto.KeyType;
|
import org.keycloak.crypto.KeyType;
|
||||||
import org.keycloak.crypto.KeyUse;
|
import org.keycloak.crypto.KeyUse;
|
||||||
|
|
||||||
import java.math.BigInteger;
|
|
||||||
import java.security.Key;
|
import java.security.Key;
|
||||||
import java.security.PublicKey;
|
import java.security.PublicKey;
|
||||||
import java.security.cert.X509Certificate;
|
import java.security.cert.X509Certificate;
|
||||||
import java.security.interfaces.ECPublicKey;
|
import java.security.interfaces.ECPublicKey;
|
||||||
import java.security.interfaces.RSAPublicKey;
|
import java.security.interfaces.RSAPublicKey;
|
||||||
|
|
||||||
|
import static org.keycloak.jose.jwk.JWKUtil.toIntegerBytes;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author <a href="mailto:sthorger@redhat.com">Stian Thorgersen</a>
|
* @author <a href="mailto:sthorger@redhat.com">Stian Thorgersen</a>
|
||||||
*/
|
*/
|
||||||
|
@ -113,8 +114,6 @@ public class JWKBuilder {
|
||||||
|
|
||||||
String kid = this.kid != null ? this.kid : KeyUtils.createKeyId(key);
|
String kid = this.kid != null ? this.kid : KeyUtils.createKeyId(key);
|
||||||
int fieldSize = ecKey.getParams().getCurve().getField().getFieldSize();
|
int fieldSize = ecKey.getParams().getCurve().getField().getFieldSize();
|
||||||
BigInteger affineX = ecKey.getW().getAffineX();
|
|
||||||
BigInteger affineY = ecKey.getW().getAffineY();
|
|
||||||
|
|
||||||
k.setKeyId(kid);
|
k.setKeyId(kid);
|
||||||
k.setKeyType(KeyType.EC);
|
k.setKeyType(KeyType.EC);
|
||||||
|
@ -126,32 +125,4 @@ public class JWKBuilder {
|
||||||
|
|
||||||
return k;
|
return k;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Copied from org.apache.commons.codec.binary.Base64
|
|
||||||
*/
|
|
||||||
private static byte[] toIntegerBytes(final BigInteger bigInt) {
|
|
||||||
int bitlen = bigInt.bitLength();
|
|
||||||
// round bitlen
|
|
||||||
bitlen = ((bitlen + 7) >> 3) << 3;
|
|
||||||
final byte[] bigBytes = bigInt.toByteArray();
|
|
||||||
|
|
||||||
if (((bigInt.bitLength() % 8) != 0) && (((bigInt.bitLength() / 8) + 1) == (bitlen / 8))) {
|
|
||||||
return bigBytes;
|
|
||||||
}
|
|
||||||
// set up params for copying everything but sign bit
|
|
||||||
int startSrc = 0;
|
|
||||||
int len = bigBytes.length;
|
|
||||||
|
|
||||||
// if bigInt is exactly byte-aligned, just skip signbit in copy
|
|
||||||
if ((bigInt.bitLength() % 8) == 0) {
|
|
||||||
startSrc = 1;
|
|
||||||
len--;
|
|
||||||
}
|
|
||||||
final int startDst = bitlen / 8 - len; // to pad w/ nulls as per spec
|
|
||||||
final byte[] resizedBytes = new byte[bitlen / 8];
|
|
||||||
System.arraycopy(bigBytes, startSrc, resizedBytes, startDst, len);
|
|
||||||
return resizedBytes;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
52
core/src/main/java/org/keycloak/jose/jwk/JWKUtil.java
Normal file
52
core/src/main/java/org/keycloak/jose/jwk/JWKUtil.java
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2021 Red Hat, Inc. and/or its affiliates
|
||||||
|
* and other contributors as indicated by the @author tags.
|
||||||
|
*
|
||||||
|
* 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.keycloak.jose.jwk;
|
||||||
|
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public class JWKUtil {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert BigInteger to 64-byte integer array
|
||||||
|
*
|
||||||
|
* Copied from org.apache.commons.codec.binary.Base64
|
||||||
|
*/
|
||||||
|
public static byte[] toIntegerBytes(final BigInteger bigInt) {
|
||||||
|
int bitlen = bigInt.bitLength();
|
||||||
|
// round bitlen
|
||||||
|
bitlen = ((bitlen + 7) >> 3) << 3;
|
||||||
|
final byte[] bigBytes = bigInt.toByteArray();
|
||||||
|
|
||||||
|
if (((bigInt.bitLength() % 8) != 0) && (((bigInt.bitLength() / 8) + 1) == (bitlen / 8))) {
|
||||||
|
return bigBytes;
|
||||||
|
}
|
||||||
|
// set up params for copying everything but sign bit
|
||||||
|
int startSrc = 0;
|
||||||
|
int len = bigBytes.length;
|
||||||
|
|
||||||
|
// if bigInt is exactly byte-aligned, just skip signbit in copy
|
||||||
|
if ((bigInt.bitLength() % 8) == 0) {
|
||||||
|
startSrc = 1;
|
||||||
|
len--;
|
||||||
|
}
|
||||||
|
final int startDst = bitlen / 8 - len; // to pad w/ nulls as per spec
|
||||||
|
final byte[] resizedBytes = new byte[bitlen / 8];
|
||||||
|
System.arraycopy(bigBytes, startSrc, resizedBytes, startDst, len);
|
||||||
|
return resizedBytes;
|
||||||
|
}
|
||||||
|
}
|
|
@ -35,10 +35,15 @@ import java.security.SecureRandom;
|
||||||
import java.security.Security;
|
import java.security.Security;
|
||||||
import java.security.Signature;
|
import java.security.Signature;
|
||||||
import java.security.cert.X509Certificate;
|
import java.security.cert.X509Certificate;
|
||||||
|
import java.security.interfaces.ECPublicKey;
|
||||||
import java.security.spec.ECGenParameterSpec;
|
import java.security.spec.ECGenParameterSpec;
|
||||||
|
import java.security.spec.ECPoint;
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.assertArrayEquals;
|
||||||
import static org.keycloak.common.util.CertificateUtils.*;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.keycloak.common.util.CertificateUtils.generateV1SelfSignedCertificate;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author <a href="mailto:sthorger@redhat.com">Stian Thorgersen</a>
|
* @author <a href="mailto:sthorger@redhat.com">Stian Thorgersen</a>
|
||||||
|
@ -150,8 +155,15 @@ public class JWKTest {
|
||||||
byte[] xBytes = Base64Url.decode(ecJwk.getX());
|
byte[] xBytes = Base64Url.decode(ecJwk.getX());
|
||||||
byte[] yBytes = Base64Url.decode(ecJwk.getY());
|
byte[] yBytes = Base64Url.decode(ecJwk.getY());
|
||||||
|
|
||||||
assertEquals(256/8, xBytes.length);
|
assertTrue(publicKey instanceof ECPublicKey);
|
||||||
assertEquals(256/8, yBytes.length);
|
ECPoint ecPoint = ((ECPublicKey) publicKey).getW();
|
||||||
|
assertNotNull(ecPoint);
|
||||||
|
|
||||||
|
int lengthAffineX = JWKUtil.toIntegerBytes(ecPoint.getAffineX()).length;
|
||||||
|
int lengthAffineY = JWKUtil.toIntegerBytes(ecPoint.getAffineY()).length;
|
||||||
|
|
||||||
|
assertEquals(lengthAffineX, xBytes.length);
|
||||||
|
assertEquals(lengthAffineY, yBytes.length);
|
||||||
|
|
||||||
String jwkJson = JsonSerialization.writeValueAsString(jwk);
|
String jwkJson = JsonSerialization.writeValueAsString(jwk);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue