PDNDResolver.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.pdd.config;
import java.io.Serializable;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.openspcoop2.core.commons.CoreException;
import org.openspcoop2.core.constants.Costanti;
import org.openspcoop2.core.id.IDSoggetto;
import org.openspcoop2.core.transazioni.utils.CredenzialiMittente;
import org.openspcoop2.pdd.core.autenticazione.GestoreAutenticazione;
import org.openspcoop2.pdd.core.keystore.KeystoreException;
import org.openspcoop2.pdd.core.keystore.RemoteStoreProvider;
import org.openspcoop2.pdd.core.token.InformazioniToken;
import org.openspcoop2.pdd.core.token.TokenUtilities;
import org.openspcoop2.pdd.core.transazioni.Transaction;
import org.openspcoop2.pdd.core.transazioni.TransactionContext;
import org.openspcoop2.pdd.core.transazioni.TransactionDeletedException;
import org.openspcoop2.pdd.core.transazioni.TransactionNotExistsException;
import org.openspcoop2.protocol.engine.SecurityTokenUtilities;
import org.openspcoop2.protocol.sdk.Context;
import org.openspcoop2.protocol.sdk.PDNDTokenInfo;
import org.openspcoop2.protocol.sdk.PDNDTokenInfoDetails;
import org.openspcoop2.protocol.sdk.ProtocolException;
import org.openspcoop2.protocol.sdk.SecurityToken;
import org.openspcoop2.protocol.sdk.state.RequestInfo;
import org.openspcoop2.security.SecurityException;
import org.openspcoop2.security.keystore.cache.GestoreKeystoreCache;
import org.openspcoop2.utils.UtilsException;
import org.openspcoop2.utils.certificate.remote.RemoteKeyType;
import org.openspcoop2.utils.certificate.remote.RemoteStoreClientInfo;
import org.openspcoop2.utils.certificate.remote.RemoteStoreConfig;
import org.openspcoop2.utils.json.JSONUtils;
import com.fasterxml.jackson.databind.JsonNode;
/**
* PDNDResolver
*
* @author Poli Andrea (poli@link.it)
* @author $Author$
* @version $Rev$, $Date$
*/
public class PDNDResolver {
private Context context;
private List<RemoteStoreConfig> remoteStores;
public PDNDResolver(Context context, List<RemoteStoreConfig> remoteStores) {
this.context = context;
this.remoteStores = remoteStores;
}
public boolean isRemoteStore(String name) {
return isRemoteStore(name, this.remoteStores);
}
public static boolean isRemoteStore(String name, List<RemoteStoreConfig> remoteStores) {
for (RemoteStoreConfig rsc : remoteStores) {
if(name.equals(rsc.getStoreName())) {
return true;
}
}
return false;
}
public RemoteStoreConfig getRemoteStoreConfig(String name, IDSoggetto idDominio) throws ProtocolException {
return getRemoteStoreConfig(name, idDominio, this.remoteStores);
}
public static RemoteStoreConfig getRemoteStoreConfig(String name, IDSoggetto idDominio, List<RemoteStoreConfig> remoteStores) throws ProtocolException {
for (RemoteStoreConfig rsc : remoteStores) {
if(name.equals(rsc.getStoreName())) {
if(rsc.isMultitenant() && idDominio!=null && idDominio.getNome()!=null) {
try {
return rsc.newInstanceMultitenant(idDominio.getNome());
}catch(Exception e){
throw new ProtocolException(e.getMessage(),e);
}
}
return rsc;
}
}
return null;
}
public RemoteStoreConfig getRemoteStoreConfigByTokenPolicy(String name, IDSoggetto idDominio) throws ProtocolException {
return getRemoteStoreConfigByTokenPolicy(name, idDominio, this.remoteStores);
}
public static RemoteStoreConfig getRemoteStoreConfigByTokenPolicy(String name, IDSoggetto idDominio, List<RemoteStoreConfig> remoteStores) throws ProtocolException {
for (RemoteStoreConfig rsc : remoteStores) {
if(name.equals(rsc.getTokenPolicy())) {
if(rsc.isMultitenant() && idDominio!=null && idDominio.getNome()!=null) {
try {
return rsc.newInstanceMultitenant(idDominio.getNome());
}catch(Exception e){
throw new ProtocolException(e.getMessage(),e);
}
}
return rsc;
}
}
return null;
}
public RemoteStoreConfig enrichTokenInfo(RequestInfo requestInfo, IDSoggetto idSoggetto,
InformazioniToken informazioniToken, SecurityToken securityTokenForContext) throws ProtocolException {
return enrichTokenInfo(requestInfo, false, false, idSoggetto,
informazioniToken, securityTokenForContext);
}
public RemoteStoreConfig enrichTokenInfo(RequestInfo requestInfo, boolean sicurezzaMessaggio, boolean sicurezzaAudit, IDSoggetto idSoggetto) throws ProtocolException {
Object oInformazioniTokenNormalizzate = null;
if(this.context!=null) {
oInformazioniTokenNormalizzate = this.context.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_INFORMAZIONI_NORMALIZZATE);
}
InformazioniToken informazioniTokenNormalizzate = null;
if(oInformazioniTokenNormalizzate instanceof InformazioniToken) {
informazioniTokenNormalizzate = (InformazioniToken) oInformazioniTokenNormalizzate;
}
SecurityToken securityTokenForContext = SecurityTokenUtilities.readSecurityToken(this.context);
return enrichTokenInfo(requestInfo, sicurezzaMessaggio, sicurezzaAudit, idSoggetto,
informazioniTokenNormalizzate, securityTokenForContext);
}
private RemoteStoreConfig enrichTokenInfo(RequestInfo requestInfo, boolean sicurezzaMessaggio, boolean sicurezzaAudit, IDSoggetto idSoggetto,
InformazioniToken informazioniToken, SecurityToken securityTokenForContext) throws ProtocolException {
OpenSPCoop2Properties op2Properties = OpenSPCoop2Properties.getInstance();
RemoteStoreConfig rsc = null;
try {
if(op2Properties.isGestoreChiaviPDNDclientInfoEnabled()) {
rsc = getRemoteStoreConfig(idSoggetto);
if(rsc==null) {
return rsc;
}
String clientId = null;
if(informazioniToken!=null) {
clientId = informazioniToken.getClientId();
}
if(clientId==null) {
return rsc;
}
// NOTA: il kid DEVE essere preso dall'eventuale token di integrità, poichè il kid nell'access token è sempre uguale ed è quello della PDND
String kid = readKid(sicurezzaMessaggio, sicurezzaAudit, securityTokenForContext, clientId);
enrichTokenInfo(requestInfo, idSoggetto,
informazioniToken, securityTokenForContext,
rsc, clientId, kid);
}
}catch(Exception e) {
throw new ProtocolException(e.getMessage(),e);
}
return rsc;
}
private void enrichTokenInfo(RequestInfo requestInfo, IDSoggetto idSoggetto,
InformazioniToken informazioniToken, SecurityToken securityTokenForContext,
RemoteStoreConfig rsc, String clientId, String kid) throws KeystoreException, SecurityException, UtilsException, TransactionNotExistsException, CoreException, TransactionDeletedException {
/**if(kid.startsWith(REMOTE_STORE_KEY_KID_STARTS_WITH_CLIENT_ID)) {*/
Object oInfoPDND = null;
if(this.context!=null) {
oInfoPDND = this.context.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_INFORMAZIONI_PDND_CLIENT_READ);
}
if(oInfoPDND instanceof String) {
String s = (String) oInfoPDND;
if("true".equals(s)) {
// chiamata fatta dalla validazione semantica dopo che la raccolta delle informazioni sulla PDND è già stata chiamata in seguito alla validazione del token
// poiche' il kid non e' presente, e' inutile ri-effettuare la medesima invocazione
return;
}
}
/**}*/
enrichTokenInfo(securityTokenForContext, informazioniToken, requestInfo, rsc,
kid, clientId);
if(kid.startsWith(REMOTE_STORE_KEY_KID_STARTS_WITH_CLIENT_ID)) {
this.context.put(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_INFORMAZIONI_PDND_CLIENT_READ, "true");
}
updateCredenzialiTokenPDND(requestInfo, idSoggetto, informazioniToken);
}
private void updateCredenzialiTokenPDND(RequestInfo requestInfo, IDSoggetto idSoggetto, InformazioniToken informazioniToken) throws TransactionNotExistsException, CoreException, TransactionDeletedException {
String idTransazione = null;
if(this.context!=null) {
idTransazione = (String) this.context.getObject(Costanti.ID_TRANSAZIONE);
}
Transaction transaction = TransactionContext.getTransaction(idTransazione);
CredenzialiMittente credenzialiMittente = transaction.getCredenzialiMittente();
if(credenzialiMittente==null) {
credenzialiMittente = new CredenzialiMittente();
transaction.setCredenzialiMittente(credenzialiMittente);
}
try {
GestoreAutenticazione.updateCredenzialiTokenPDND(idSoggetto, "ModIValidator", idTransazione,
informazioniToken, credenzialiMittente,
null, "ModIValidator.credenzialiPDND", requestInfo,
this.context);
}catch(Exception e) {
throw new CoreException(e.getMessage(),e);
}
}
private RemoteStoreConfig getRemoteStoreConfig(IDSoggetto idSoggetto) throws ProtocolException {
Object oTokenPolicy = null;
if(this.context!=null) {
oTokenPolicy = this.context.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_POLICY);
}
String tokenPolicy = null;
if(oTokenPolicy instanceof String) {
tokenPolicy = (String) oTokenPolicy;
}
if(tokenPolicy==null) {
return null;
}
return this.getRemoteStoreConfigByTokenPolicy(tokenPolicy, idSoggetto);
}
public static final String REMOTE_STORE_KEY_KID_STARTS_WITH_CLIENT_ID = "ClientId--";
private String readKid(boolean sicurezzaMessaggio, boolean sicurezzaAudit, SecurityToken securityTokenForContext, String clientId) throws UtilsException {
// NOTA: il kid DEVE essere preso dall'eventuale token di integrità, poichè il kid nell'access token è sempre uguale ed è quello della PDND
String kid = null;
if(sicurezzaMessaggio) {
kid = readKidFromTokenIntegrity(securityTokenForContext);
}
if(kid==null && sicurezzaAudit) {
kid = readKidFromTokenAudit(securityTokenForContext);
}
if(kid==null) {
// Altrimenti utilizzo la struttura dati per ospitare le informazioni sul clientId
kid = REMOTE_STORE_KEY_KID_STARTS_WITH_CLIENT_ID+clientId;
}
return kid;
}
private String readKidFromTokenIntegrity(SecurityToken securityTokenForContext) throws UtilsException {
String kid = null;
if(securityTokenForContext!=null && securityTokenForContext.getIntegrity()!=null) {
kid = securityTokenForContext.getIntegrity().getKid();
if(kid==null) {
kid = securityTokenForContext.getIntegrity().getHeaderClaim("kid");
}
}
return kid;
}
private String readKidFromTokenAudit(SecurityToken securityTokenForContext) throws UtilsException {
String kid = null;
if(securityTokenForContext!=null && securityTokenForContext.getAudit()!=null) {
kid = securityTokenForContext.getAudit().getKid();
if(kid==null) {
kid = securityTokenForContext.getAudit().getHeaderClaim("kid");
}
}
return kid;
}
private void enrichTokenInfo(SecurityToken securityTokenForContext, InformazioniToken informazioniTokenNormalizzate, RequestInfo requestInfo, RemoteStoreConfig rsc,
String kid, String clientId) throws KeystoreException, SecurityException, UtilsException {
RemoteKeyType keyType = RemoteKeyType.JWK; // ignored
RemoteStoreProvider remoteStoreProvider = new RemoteStoreProvider(requestInfo, keyType);
RemoteStoreClientInfo rsci = GestoreKeystoreCache.getRemoteStoreClientInfo(requestInfo, kid, clientId, rsc, remoteStoreProvider, this.context);
if(rsci!=null &&
(rsci.getClientDetails()!=null || rsci.getOrganizationId()!=null || rsci.getOrganizationDetails()!=null)
){
if(informazioniTokenNormalizzate.getPdnd()==null) {
informazioniTokenNormalizzate.setPdnd(new HashMap<>());
}
if(rsci.getClientDetails()!=null) {
JSONUtils jsonUtils = JSONUtils.getInstance();
if(jsonUtils.isJson(rsci.getClientDetails())) {
JsonNode root = jsonUtils.getAsNode(rsci.getClientDetails());
PDNDTokenInfoDetails info = new PDNDTokenInfoDetails();
info.setId(rsci.getClientId());
info.setDetails(rsci.getClientDetails());
enrichTokenInfoAddInClaims(jsonUtils, securityTokenForContext, informazioniTokenNormalizzate, root, PDNDTokenInfo.CLIENT_INFO, info);
}
}
if(rsci.getOrganizationDetails()!=null) {
JSONUtils jsonUtils = JSONUtils.getInstance();
if(jsonUtils.isJson(rsci.getOrganizationDetails())) {
JsonNode root = jsonUtils.getAsNode(rsci.getOrganizationDetails());
PDNDTokenInfoDetails info = new PDNDTokenInfoDetails();
info.setId(rsci.getOrganizationId());
info.setDetails(rsci.getOrganizationDetails());
enrichTokenInfoAddInClaims(jsonUtils, securityTokenForContext, informazioniTokenNormalizzate, root, PDNDTokenInfo.ORGANIZATION_INFO, info);
}
}
}
}
private void enrichTokenInfoAddInClaims(JSONUtils jsonUtils, SecurityToken securityTokenForContext, InformazioniToken informazioniTokenNormalizzate, JsonNode root, String type,
PDNDTokenInfoDetails info) {
Map<String, Serializable> readClaims = jsonUtils.convertToSimpleMap(root);
if(!readClaims.isEmpty()) {
enrichTokenInfoAddInClaims(securityTokenForContext, informazioniTokenNormalizzate,
type, info,
readClaims);
}
}
private void enrichTokenInfoAddInClaims(SecurityToken securityTokenForContext, InformazioniToken informazioniTokenNormalizzate,
String type, PDNDTokenInfoDetails info,
Map<String, Serializable> readClaims) {
informazioniTokenNormalizzate.getPdnd().put(type,TokenUtilities.toHashMapSerializable(readClaims));
String prefix = PDNDTokenInfo.TOKEN_INFO_PREFIX_PDND+type+".";
Map<String, Serializable> readClaimsSerializable = new HashMap<>();
if(informazioniTokenNormalizzate.getClaims()!=null) {
for (Map.Entry<String,Serializable> entry : readClaims.entrySet()) {
String key = prefix+entry.getKey();
if(!informazioniTokenNormalizzate.getClaims().containsKey(key)) {
informazioniTokenNormalizzate.getClaims().put(key, entry.getValue());
}
if(entry.getValue() instanceof Serializable) {
readClaimsSerializable.put(entry.getKey(), entry.getValue());
}
}
}
info.setClaims(readClaimsSerializable);
if(securityTokenForContext!=null) {
if(securityTokenForContext.getPdnd()==null) {
securityTokenForContext.setPdnd(new PDNDTokenInfo());
}
securityTokenForContext.getPdnd().setInfo(type, info);
}
}
}