JWK.java
/*
* GovWay - A customizable API Gateway
* https://govway.org
*
* Copyright (c) 2005-2025 Link.it srl (https://link.it).
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3, as published by
* the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
package org.openspcoop2.utils.certificate;
import java.security.PrivateKey;
import java.security.PublicKey;
import org.apache.cxf.rs.security.jose.jwk.JsonWebKey;
import org.apache.cxf.rs.security.jose.jwk.JwkReaderWriter;
import org.openspcoop2.utils.UtilsException;
import org.openspcoop2.utils.UtilsRuntimeException;
import org.openspcoop2.utils.json.JSONUtils;
import com.fasterxml.jackson.databind.JsonNode;
import com.nimbusds.jose.Algorithm;
import com.nimbusds.jose.jwk.KeyUse;
import com.nimbusds.jose.jwk.OctetSequenceKey;
import com.nimbusds.jose.jwk.RSAKey;
/**
* JWK
*
* @author Poli Andrea (apoli@link.it)
* @author $Author$
* @version $Rev$, $Date$
*/
public class JWK {
private JwkReaderWriter engineCxf = new JwkReaderWriter();
private String jwkJson;
private String jwkJsonPretty;
private JsonWebKey jwkCxf;
private com.nimbusds.jose.jwk.JWK jwkNimbusds;
private JsonNode jwkNode;
public JWK(String json) {
this.jwkJson = json;
}
public JWK(JsonWebKey jwk) {
this.jwkCxf = jwk;
}
public JWK(com.nimbusds.jose.jwk.JWK jwk) {
this.jwkNimbusds = jwk;
}
public JWK(KeyStore keystore, String alias) throws UtilsException {
this(keystore, alias, null, null, true);
}
public JWK(KeyStore keystore, String alias, KeyUse use) throws UtilsException {
this(keystore, alias, null, use, true);
}
public JWK(KeyStore keystore, String alias, boolean kid) throws UtilsException {
this(keystore, alias, null, null, kid);
}
public JWK(KeyStore keystore, String alias, KeyUse use, boolean kid) throws UtilsException {
this(keystore, alias, null, use, kid);
}
public JWK(KeyStore keystore, String alias, String passwordPrivateKey) throws UtilsException {
this(keystore, alias, passwordPrivateKey, null, true);
}
public JWK(KeyStore keystore, String alias, String passwordPrivateKey, KeyUse use) throws UtilsException {
this(keystore, alias, passwordPrivateKey, use, true);
}
public JWK(KeyStore keystore, String alias, String passwordPrivateKey, boolean kid) throws UtilsException {
this(keystore, alias, passwordPrivateKey, null, kid);
}
public JWK(KeyStore keystore, String alias, String passwordPrivateKey, KeyUse use, boolean kid) throws UtilsException {
try {
if(!keystore.existsAlias(alias)) {
throw new UtilsException("Alias '"+alias+"' undefined");
}
PublicKey publicKey = keystore.getPublicKey(alias);
if(publicKey instanceof java.security.interfaces.RSAPublicKey) {
PrivateKey privateKey = null;
if(passwordPrivateKey!=null) {
privateKey = keystore.getPrivateKey(alias, passwordPrivateKey);
}
String aliasP = alias;
if(!kid) {
aliasP = null;
}
initEngine(publicKey, privateKey, aliasP, use);
}
else {
throw new UtilsException("Unsupported type '"+publicKey.getClass().getName()+"'");
}
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
public JWK(PublicKey publicKey) throws UtilsException {
this(publicKey, null, null, null);
}
public JWK(PublicKey publicKey, String kid) throws UtilsException {
this(publicKey, null, kid, null);
}
public JWK(PublicKey publicKey, KeyUse use) throws UtilsException {
this(publicKey, null, null, use);
}
public JWK(PublicKey publicKey, String kid, KeyUse use) throws UtilsException {
this(publicKey, null, kid, use);
}
public JWK(PublicKey publicKey, PrivateKey privateKey) throws UtilsException {
this(publicKey, privateKey, null, null);
}
public JWK(PublicKey publicKey, PrivateKey privateKey, String kid) throws UtilsException {
this(publicKey, privateKey, kid, null);
}
public JWK(PublicKey publicKey, PrivateKey privateKey, KeyUse use) throws UtilsException {
this(publicKey, privateKey, null, use);
}
public JWK(PublicKey publicKey, PrivateKey privateKey, String kid, KeyUse use) throws UtilsException {
initEngine(publicKey, privateKey, kid, use);
}
private void initEngine(PublicKey publicKey, PrivateKey privateKey, String kid, KeyUse use) throws UtilsException {
try {
if(publicKey instanceof java.security.interfaces.RSAPublicKey) {
java.security.interfaces.RSAPublicKey p = (java.security.interfaces.RSAPublicKey) publicKey;
RSAKey.Builder builder = new RSAKey.Builder(p);
if(privateKey!=null) {
builder.privateKey(privateKey);
}
if(kid!=null) {
builder.keyID(kid);
}
if(use!=null) {
builder.keyUse(use);
}
this.jwkNimbusds = builder.build();
}
else {
if(publicKey==null) {
throw new UtilsException("PublicKey undefined");
}
else {
throw new UtilsException("Unsupported type '"+publicKey.getClass().getName()+"'");
}
}
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
public JWK(javax.crypto.SecretKey secretKey) throws UtilsException {
this(secretKey, null, null, null);
}
public JWK(javax.crypto.SecretKey secretKey, String kid) throws UtilsException {
this(secretKey, kid, null, null);
}
public JWK(javax.crypto.SecretKey secretKey, String kid, String algorithm) throws UtilsException {
this(secretKey, kid, null, algorithm);
}
public JWK(javax.crypto.SecretKey secretKey, KeyUse use) throws UtilsException {
this(secretKey, null, use, null);
}
public JWK(javax.crypto.SecretKey secretKey, KeyUse use, String algorithm) throws UtilsException {
this(secretKey, null, use, algorithm);
}
public JWK(javax.crypto.SecretKey secretKey, String kid, KeyUse use) throws UtilsException {
this(secretKey, kid, use, null);
}
public JWK(javax.crypto.SecretKey secretKey, String kid, KeyUse use, String algorithm) throws UtilsException {
try {
OctetSequenceKey.Builder builder = new OctetSequenceKey.Builder(secretKey);
if(algorithm!=null) {
builder = builder.algorithm(Algorithm.parse(algorithm));
}
if(kid!=null) {
builder = builder.keyID(kid);
}
if(use!=null) {
builder = builder.keyUse(use);
}
this.jwkNimbusds = builder.build();
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
private synchronized void initCxf() throws UtilsException {
if(this.jwkCxf==null) {
if(this.jwkJson==null){
throw new UtilsException("Json not defined");
}
this.jwkCxf = this.engineCxf.jsonToJwk(this.jwkJson);
}
}
public JsonWebKey getJsonWebKey() throws UtilsException {
if(this.jwkCxf==null) {
this.initCxf();
}
return this.jwkCxf;
}
private synchronized void initNimbusds() throws UtilsException {
if(this.jwkNimbusds==null) {
if(this.jwkJson==null){
throw new UtilsException("Json not defined");
}
try {
this.jwkNimbusds = RSAKey.parse(this.jwkJson);
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
}
public com.nimbusds.jose.jwk.JWK getJWK() throws UtilsException {
if(this.jwkNimbusds==null) {
this.initNimbusds();
}
return this.jwkNimbusds;
}
private synchronized void initJson() throws UtilsException {
if(this.jwkJson==null) {
if(this.jwkCxf==null && this.jwkNimbusds==null){
throw new UtilsException("JWK not defined");
}
if(this.jwkCxf!=null) {
try {
this.jwkJson = this.engineCxf.jwkToJson(this.jwkCxf);
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
else {
try {
this.jwkJson = this.jwkNimbusds.toJSONString();
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
}
}
public String getJson() throws UtilsException {
if(this.jwkJson==null) {
this.initJson();
}
return this.jwkJson;
}
private synchronized void initJsonPretty() throws UtilsException {
if(this.jwkJsonPretty==null) {
try {
if(this.jwkNode==null) {
initNode();
}
this.jwkJsonPretty = JSONUtils.getInstance(true).toString(this.jwkNode);
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
}
public String getJsonPretty() throws UtilsException {
if(this.jwkJsonPretty==null) {
this.initJsonPretty();
}
return this.jwkJsonPretty;
}
private synchronized void initNode() throws UtilsException {
if(this.jwkNode==null) {
try {
if(this.jwkJson==null) {
initJson();
}
this.jwkNode = JSONUtils.getInstance().getAsNode(this.jwkJson);
}catch(Exception e) {
throw new UtilsException(e.getMessage(),e);
}
}
}
public JsonNode getNode() throws UtilsException {
if(this.jwkNode==null) {
this.initNode();
}
return this.jwkNode;
}
@Override
public String toString() {
try {
return this.getJsonPretty();
}catch(Exception e) {
throw new UtilsRuntimeException(e.getMessage(),e);
}
}
}