JwtHeaders.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.security;

import java.net.URI;
import java.security.cert.X509Certificate;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;

import org.apache.cxf.common.util.Base64UrlUtility;
import org.apache.cxf.rs.security.jose.common.KeyManagementUtils;
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.JwkUtils;
import org.apache.cxf.rt.security.crypto.MessageDigestUtils;
import org.openspcoop2.utils.UtilsException;

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

	public static final String JWT_HDR_ALG = "alg"; // (Algorithm) Header Parameter
	public static final String JWT_HDR_JKU = "jku"; //  (JWK Set URL) Header Parameter
	public static final String JWT_HDR_JWK = "jwk"; // (JSON Web Key) Header Parameter
	public static final String JWT_HDR_KID = "kid"; // (Key ID) Header Parameter
	public static final String JWT_HDR_X5U = "x5u"; // (X.509 URL) Header Parameter
	public static final String JWT_HDR_X5C = "x5c"; // (X.509 Certificate Chain) Header Parameter
	public static final String JWT_HDR_X5T = "x5t"; // (X.509 Certificate SHA-1 Thumbprint) Header Parameter
	public static final String JWT_HDR_X5t_S256 = "x5t#S256"; // (X.509 Certificate SHA-256 Thumbprint) Header Parameter
	public static final String JWT_HDR_TYP = "typ"; // (Type) Header Parameter
	public static final String JWT_HDR_CTY = "cty"; // (Content Type) Header Parameter
	public static final String JWT_HDR_CRIT = "crit"; // (Critical) Header Parameter
	public static final String JWT_HDR_ENC = "enc"; // (Encryption Algorithm) Header Parameter [solo in jwe]
	public static final String JWT_HDR_ZIP = "zip"; // (Compression Algorithm) Header Parameter [solo in jwe]
	 
	private String type;
	private String contentType;
	private String kid; 
	private List<String> criticalHeaders = new ArrayList<>();
	private URI x509Url;
	private List<X509Certificate> x509c = new ArrayList<>(); // i certificati servono anche per sha1 e sha256, il field addX5C serve quindi per capire se poi far uscire anche X5C
	private boolean addX5C = false;
	boolean x509IncludeCertSha1 = false;
	boolean x509IncludeCertSha256 = false;
	private URI jwkUrl;
	private JsonWebKey jwKey;
	private HashMap<String, String> extensions = new HashMap<>();
	
	public void setType(String type) {
		this.type = type;
	}
	public void setContentType(String contentType) {
		this.contentType = contentType;
	}
	public String getKid() {
		return this.kid;
	}
	public void addCriticalHeader(String hdr) {
		this.criticalHeaders.add(hdr);
	}
	public void setX509Url(URI x509Url) {
		this.x509Url = x509Url;
	}
	public void addX509cert(X509Certificate x509c) {
		this.x509c.add(x509c);
	}
	public void setAddX5C(boolean addX5C) {
		this.addX5C = addX5C;
	}
	public boolean isX509IncludeCertSha1() {
		return this.x509IncludeCertSha1;
	}
	public boolean isX509IncludeCertSha256() {
		return this.x509IncludeCertSha256;
	}
	public void setJwkUrl(URI jwkUrl) {
		this.jwkUrl = jwkUrl;
	}
	public void setJwKey(JsonWebKey jwKey) {
		this.jwKey = jwKey;
	}
	public void setJwKey(JsonWebKeys jsonWebKeys, String alias) throws UtilsException {
		this.jwKey = JsonUtils.readKey(jsonWebKeys, alias);
	}
	public void addExtension(String hdr, String value) {
		this.extensions.put(hdr, value);
	}
	
	public String getType() {
		return this.type;
	}
	public String getContentType() {
		return this.contentType;
	}
	public void setKid(String kid) {
		this.kid = kid;
	}
	public List<String> getCriticalHeaders() {
		return this.criticalHeaders;
	}
	public URI getX509Url() {
		return this.x509Url;
	}
	public boolean isAddX5C() {
		return this.addX5C;
	}
	public List<X509Certificate> getX509c() {
		return this.x509c;
	}
	public void setX509IncludeCertSha1(boolean includeCertSha1) {
		this.x509IncludeCertSha1 = includeCertSha1;
	}
	public void setX509IncludeCertSha256(boolean includeCertSha256) {
		this.x509IncludeCertSha256 = includeCertSha256;
	}
	public URI getJwkUrl() {
		return this.jwkUrl;
	}
	public JsonWebKey getJwKey() {
		return this.jwKey;
	}
	public HashMap<String, String> getExtensions() {
		return this.extensions;
	}
	
	public List<String> headers(){
		List<String> list = new ArrayList<>();
		if(this.type!=null) {
			list.add(JWT_HDR_TYP);
		}
		if(this.contentType!=null) {
			list.add(JWT_HDR_CTY);
		}
		if(this.kid!=null) {
			list.add(JWT_HDR_KID);
		}
		if(this.criticalHeaders!=null && !this.criticalHeaders.isEmpty()) {
			list.add(JWT_HDR_CRIT);
		}
		if(this.x509Url!=null) {
			list.add(JWT_HDR_X5U);
		}
		if(this.x509c!=null && !this.x509c.isEmpty()) {
			// fix: lo aggiungo solo se non c'è la url. Nell'oggetto JwtHreader il certificato ho dovuto mettercelo per creare i sha
			//if(!list.contains(JWT_HDR_X5U)) {
			// il fix era errato, aggiunto field apposito 'addX5C'
			if(this.addX5C) {
				list.add(JWT_HDR_X5C);
			}
		}
		if(this.x509IncludeCertSha1 && this.x509c!=null && !this.x509c.isEmpty()) {
			list.add(JWT_HDR_X5T);
		}
		if(this.x509IncludeCertSha256) {
			list.add(JWT_HDR_X5t_S256);
		}
		if(this.jwkUrl!=null) {
			list.add(JWT_HDR_JKU);
		}
		if(this.jwKey!=null) {
			list.add(JWT_HDR_JWK);
		}
		if(this.extensions!=null && !this.extensions.isEmpty()) {
			Iterator<String> hdrIt = this.extensions.keySet().iterator();
			while (hdrIt.hasNext()) {
				String hdr = (String) hdrIt.next();
				list.add(hdr);
			}
		}
		return list;
	}
	
	public void fillJwsHeaders(org.apache.cxf.rs.security.jose.common.JoseHeaders hdrs, boolean forceOverride, String algorithm) throws Exception {
		if(this.type!=null) {
			if(!hdrs.containsHeader(JWT_HDR_TYP) || forceOverride) {
				hdrs.setHeader(JWT_HDR_TYP, this.type);
			}
		}
		if(this.contentType!=null) {
			if(!hdrs.containsHeader(JWT_HDR_CTY) || forceOverride) {
				hdrs.setContentType(this.contentType);
			}
		}
		if(this.kid!=null) {
			if(!hdrs.containsHeader(JWT_HDR_KID) || forceOverride) {
				hdrs.setKeyId(this.kid);
			}
		}
		if(this.criticalHeaders!=null && !this.criticalHeaders.isEmpty()) {
			List<String> headers = new ArrayList<>();
			if(hdrs.containsHeader(JWT_HDR_CRIT)) {
				headers = hdrs.getCritical();
				if(headers==null) {
					headers = new ArrayList<>();
				}
			}
			for (String ch : this.criticalHeaders) {
				if(headers.contains(ch)==false) {
					headers.add(ch);
				}
			}
			/*
			StringBuilder bf = new StringBuilder();
			for (String ch : headers) {
				if(bf.length()>0) {
					bf.append(",");
				}
				bf.append("\"").append(ch).append("\"");
			}
			hdrs.setHeader(JWT_HDR_CRIT, "["+bf.toString()+"]");*/
			hdrs.setCritical(headers);
		}
		if(this.x509Url!=null) {
			if(!hdrs.containsHeader(JWT_HDR_X5U) || forceOverride) {
				hdrs.setX509Url(this.x509Url.toString());
			}
		}
		if(this.x509c!=null && !this.x509c.isEmpty()) {
			if(!hdrs.containsHeader(JWT_HDR_X5C) || forceOverride) {
				// fix: lo aggiungo solo se non c'è la url. Nell'oggetto JwtHreader il certificato ho dovuto mettercelo per creare i sha
				//if(!hdrs.containsHeader(JWT_HDR_X5U)) {
				// il fix era errato, aggiunto field apposito 'addX5C'
				if(this.addX5C) {
					X509Certificate[] chain = this.x509c.toArray(new X509Certificate[1]);
					hdrs.setX509Chain(KeyManagementUtils.encodeX509CertificateChain(chain));
				}
			}
		}
		if(this.x509IncludeCertSha1 && this.x509c!=null && !this.x509c.isEmpty()) {
			if(!hdrs.containsHeader(JWT_HDR_X5T) || forceOverride) {
				X509Certificate[] chain = this.x509c.toArray(new X509Certificate[1]);
				byte[] digestB = MessageDigestUtils.createDigest(chain[0].getEncoded(), MessageDigestUtils.ALGO_SHA_1);
				String digest = Base64UrlUtility.encode(digestB);
				hdrs.setX509Thumbprint(digest);
			}
		}
		if(this.x509IncludeCertSha256) {
			if(!hdrs.containsHeader(JWT_HDR_X5t_S256) || forceOverride) {
				X509Certificate[] chain = this.x509c.toArray(new X509Certificate[1]);
				byte[] digestB = MessageDigestUtils.createDigest(chain[0].getEncoded(), MessageDigestUtils.ALGO_SHA_256);
				String digest = Base64UrlUtility.encode(digestB);
				hdrs.setX509ThumbprintSHA256(digest);
			}
		}
		if(this.jwkUrl!=null) {
			if(!hdrs.containsHeader(JWT_HDR_JKU) || forceOverride) {
				hdrs.setJsonWebKeysUrl(this.jwkUrl.toString());
			}
		}
		if(this.jwKey!=null) {
			if(!hdrs.containsHeader(JWT_HDR_JWK) || forceOverride) {
				JwkUtils.includeCertChain(this.jwKey, hdrs, algorithm);
				JwkUtils.includePublicKey(this.jwKey, hdrs, algorithm);
			}
		}
		if(this.extensions!=null && !this.extensions.isEmpty()) {
			Iterator<String> hdrIt = this.extensions.keySet().iterator();
			while (hdrIt.hasNext()) {
				String hdr = (String) hdrIt.next();
				if(!hdrs.containsHeader(hdr) || forceOverride) {
					String value = this.extensions.get(hdr);
					hdrs.setHeader(hdr,value);
				}
			}
		}
	}
	
}