JWKSet.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.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import org.apache.cxf.rs.security.jose.jwk.JsonWebKey;
import org.apache.cxf.rs.security.jose.jwk.JsonWebKeys;
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;

/**	
 * JWKSet
 *
 * @author Poli Andrea (apoli@link.it)
 * @author $Author$
 * @version $Rev$, $Date$
 */
public class JWKSet {

	private JwkReaderWriter engineCxf = new JwkReaderWriter();
	private String jwkSetJson;
	private String jwkSetJsonPretty;
	private JsonWebKeys jwkSetCxf;
	private com.nimbusds.jose.jwk.JWKSet jwkSetNimbusds;
	private JsonNode jwkSetNode;
	private List<JWK> jwkSetList = new ArrayList<>();
	
	public JWKSet(String json) {
		this.jwkSetJson = json;
	}
	
	public JWKSet(JsonWebKeys jwk) {
		this.jwkSetCxf = jwk;
	}
	
	public JWKSet(com.nimbusds.jose.jwk.JWKSet jwk) {
		this.jwkSetNimbusds = jwk;
	}
	
	public JWKSet(List<JWK> list) {
		this.jwkSetList = list;
	}
	
	public JWKSet() {

	}
	
	
	public void addJwk(JWK jwk) {
		this.jwkSetList.add(jwk);
		
		// forzo rebuild
		this.jwkSetJson = null;
		this.jwkSetJsonPretty = null;
		this.jwkSetCxf = null;
		this.jwkSetNimbusds = null;
		this.jwkSetNode = null;
				
	}
	
	private synchronized void initJwks() throws UtilsException {
		if(this.jwkSetList==null || this.jwkSetList.isEmpty()) {
			if(this.getJsonWebKeys()==null && this.getJWKSet()==null){
				throw new UtilsException("JWK Set not defined");
			}
			if(this.jwkSetCxf!=null) {
				initJwksFromSetCxf();
			}
			else {
				initJwksFromSetNimbusds();
			}
		}
	}
	private synchronized void initJwksFromSetCxf() throws UtilsException {
		try {
			Iterator<JsonWebKey> it = this.jwkSetCxf.getKeys().iterator();
			while (it.hasNext()) {
				JsonWebKey jsonWebKey = it.next();
				this.jwkSetList.add(new JWK(jsonWebKey));
			}
		}catch(Exception e) {
			throw new UtilsException(e.getMessage(),e);
		}
	}
	private synchronized void initJwksFromSetNimbusds() throws UtilsException {
		try {
			Iterator<com.nimbusds.jose.jwk.JWK> it = this.jwkSetNimbusds.getKeys().iterator();
			while (it.hasNext()) {
				com.nimbusds.jose.jwk.JWK jwk = it.next();
				this.jwkSetList.add(new JWK(jwk));
			}
		}catch(Exception e) {
			throw new UtilsException(e.getMessage(),e);
		}
	}
	public List<JWK> getJwks() throws UtilsException {
		if(this.jwkSetList==null || this.jwkSetList.isEmpty()) {
			this.initJwks();
		}
		return this.jwkSetList;
	}
	
	
	
	private synchronized void initCxf() throws UtilsException {
		if(this.jwkSetCxf==null) {
			if(this.jwkSetJson==null){
				throw new UtilsException("Json not defined");
			}
			try {
				this.jwkSetCxf = this.engineCxf.jsonToJwkSet(this.jwkSetJson);
			}catch(Exception e) {
				/* Fix bug cxf che va in errore se il keystore รจ pretty print e la , tra una chiave e l'altra struttura json va a capo.
				 * Esempio:
				 * {
					  "keys" : [ 
					{
					  "kty" : "RSA",
					  "e" : "AQAB",
					  "kid" : "KID-ApplicativoBlockingIDA01",
					  "n" : "8IYnrIeYyuCCZQJvdkxL5bD5-v-L2bwAgiz4ZMJPqWqhbgsGmneyaHQIKL-ihiKRgbDT02HIDlLeJp6uXCxRu6MYVEXkpuI1Kte2xqlGzw2F-WALNkoFgaXeIrEwIugj5eeiqqBbedZdQPr7YmXHiZOSztVsQSeyRTGhIfMtvrqKUa8R4U2gFAp5wo2cgCU2Dk1gJ_B2mooxgvewi2Ea2SSuuOAJThvAEwAk1cxXwZcxVqOAjzgeKe7kPs79VgCpnGuottrhqlLtT0hPpij2T1S_r2ENZrQ9ex4hkFF8q2EpxqveKcF5bmTaNS1ezjKCthJjmq1zu-zELLFAx4bY_w"
					}
					,
					{
					  "kty" : "RSA",
					  "e" : "AQAB",
					  "kid" : "KID-ExampleServer",
					  "n" : "4OKbeAjhuWBnATnd2FjdvRAdyks05AnW5nYNRWVt2RIvBfPsASZD858hv_ts1W2uUN1EbJTSgQzgZskufBKz3KApI1Lq3F3IEH2jLBYGywpCbus6hHNCi8xN1OzRLEDp-uaZvIeP26RDjush53j9YFvVEI5Hic6thLT0zqtFhm-u1VtDH0uEZ_1S5CUspMYbZTOkl-PEz7Y77dIGk0vfhfJ3uW3g1khWQONVHA7X4XOZLKDo1rnQxzZv3l7r__h5GlHqZRopLBwqn6hDRyoeRzZfQtrl_fMp5Pgyg0Fi5hKI5o0YLnOhHzB_MJJrgoXOjirvoFBO-qkET-BAjU6BAQ"
					}
					   ]
					}
				 *  Produce il seguente errore: "String index out of range: 0"
				 * 
				 **/
				if(e.getMessage()!=null && e.getMessage().contains("String index out of range")) {
					try {
						JSONUtils jsonUtils = JSONUtils.getInstance();
						JsonNode node = jsonUtils.getAsNode(this.jwkSetJson);
						String sNonPretty = jsonUtils.toString(node);
						this.jwkSetCxf = this.engineCxf.jsonToJwkSet(sNonPretty);
					}catch(Exception eCxf) {
						// rilancio eccezione originale
						throw e;
					}
				}
			}
		}
	}
	public JsonWebKeys getJsonWebKeys() throws UtilsException {
		if(this.jwkSetCxf==null) {
			this.initCxf();
		}
		return this.jwkSetCxf;
	}
	
	private synchronized void initNimbusds() throws UtilsException {
		if(this.jwkSetNimbusds==null) {
			if(this.jwkSetJson==null){
				throw new UtilsException("Json not defined");
			}
			try {
				this.jwkSetNimbusds = com.nimbusds.jose.jwk.JWKSet.parse(this.jwkSetJson);
			}catch(Exception e) {
				throw new UtilsException(e.getMessage(),e);
			}
		}
	}
	public com.nimbusds.jose.jwk.JWKSet getJWKSet() throws UtilsException {
		if(this.jwkSetNimbusds==null) {
			this.initNimbusds();
		}
		return this.jwkSetNimbusds;
	}
	
	private synchronized void initJson() throws UtilsException {
		if(this.jwkSetJson==null) {
			if( (this.jwkSetList==null || this.jwkSetList.isEmpty()) && this.jwkSetCxf==null && this.jwkSetNimbusds==null){
				throw new UtilsException("JWK Set not defined");
			}
			if(this.jwkSetList!=null && !this.jwkSetList.isEmpty()) {
				initJsonFromJwkSetList();
			}
			else if(this.jwkSetCxf!=null) {
				initJsonFromJwkSetCxf();
			}
			else {
				initJsonFromJwkSetNimbusds();
			}
		}
	}
	private synchronized void initJsonFromJwkSetList() throws UtilsException {
		List<com.nimbusds.jose.jwk.JWK> list = new ArrayList<>();
		for (JWK jwkOp : this.jwkSetList) {
			list.add(jwkOp.getJWK());
		}
		/** NON FUNZIONA 
		com.nimbusds.jose.jwk.JWKSet set = new com.nimbusds.jose.jwk.JWKSet(list);
		this.jwkSetJson = set.toJSONObject().toString(); */
		StringBuilder sb = new StringBuilder("{\"keys\":[");
		boolean first = true;
		for (JWK jwkOp : this.jwkSetList) {
			if(!first) {
				sb.append(",");
			}
			sb.append(jwkOp.getJWK().toJSONString());
			first = false;
		}
		sb.append("]}");
		this.jwkSetJson = sb.toString();
	}
	private synchronized void initJsonFromJwkSetCxf() throws UtilsException {
		try {
			this.jwkSetJson = this.engineCxf.jwkSetToJson(this.jwkSetCxf);
		}catch(Exception e) {
			throw new UtilsException(e.getMessage(),e);
		}
	}
	private synchronized void initJsonFromJwkSetNimbusds() throws UtilsException {
		try {
			this.jwkSetJson = this.jwkSetNimbusds.toString();
		}catch(Exception e) {
			throw new UtilsException(e.getMessage(),e);
		}
	}
	public String getJson() throws UtilsException {
		if(this.jwkSetJson==null) {
			this.initJson();
		}
		return this.jwkSetJson;
	}
	
	private synchronized void initJsonPretty() throws UtilsException {
		if(this.jwkSetJsonPretty==null) {
			try {
				if(this.jwkSetNode==null) {
					initNode();
				}
				this.jwkSetJsonPretty = JSONUtils.getInstance(true).toString(this.jwkSetNode);
			}catch(Exception e) {
				throw new UtilsException(e.getMessage(),e);
			}
		}
	}
	public String getJsonPretty() throws UtilsException {
		if(this.jwkSetJsonPretty==null) {
			this.initJsonPretty();
		}
		return this.jwkSetJsonPretty;
	}
	
	private synchronized void initNode() throws UtilsException {
		if(this.jwkSetNode==null) {
			try {
				if(this.jwkSetJson==null) {
					initJson();
				}
				this.jwkSetNode = JSONUtils.getInstance().getAsNode(this.jwkSetJson);
			}catch(Exception e) {
				throw new UtilsException(e.getMessage(),e);
			}
		}
	}
	public JsonNode getNode() throws UtilsException {
		if(this.jwkSetNode==null) {
			this.initNode();
		}
		return this.jwkSetNode;
	}

	@Override
	public String toString() {
		try {
			return this.getJsonPretty();
		}catch(Exception e) {
			throw new UtilsRuntimeException(e.getMessage(),e);
		}
	}
}