GestoreAutenticazione.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.core.autenticazione;

import java.security.SecureRandom;
import java.sql.Connection;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.commons.lang.StringUtils;
import org.openspcoop2.core.commons.dao.DAOFactory;
import org.openspcoop2.core.commons.dao.DAOFactoryProperties;
import org.openspcoop2.core.config.GestioneTokenAutenticazione;
import org.openspcoop2.core.config.PortaDelegata;
import org.openspcoop2.core.config.constants.CostantiConfigurazione;
import org.openspcoop2.core.config.constants.StatoFunzionalita;
import org.openspcoop2.core.id.IDPortaApplicativa;
import org.openspcoop2.core.id.IDPortaDelegata;
import org.openspcoop2.core.id.IDServizioApplicativo;
import org.openspcoop2.core.id.IDSoggetto;
import org.openspcoop2.core.transazioni.CredenzialeMittente;
import org.openspcoop2.core.transazioni.IdCredenzialeMittente;
import org.openspcoop2.core.transazioni.dao.ICredenzialeMittenteService;
import org.openspcoop2.core.transazioni.utils.CredenzialiMittente;
import org.openspcoop2.core.transazioni.utils.TipoCredenzialeMittente;
import org.openspcoop2.core.transazioni.utils.credenziali.AbstractCredenziale;
import org.openspcoop2.core.transazioni.utils.credenziali.AbstractSearchCredenziale;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeApi;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeClientAddress;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeEventi;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeGruppi;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchApi;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchClientAddress;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchEvento;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchGruppo;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchToken;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchTokenClient;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeSearchTrasporto;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeToken;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeTokenClient;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeTokenPDND;
import org.openspcoop2.core.transazioni.utils.credenziali.CredenzialeTrasporto;
import org.openspcoop2.generic_project.expression.IPaginatedExpression;
import org.openspcoop2.generic_project.utils.ServiceManagerProperties;
import org.openspcoop2.message.OpenSPCoop2Message;
import org.openspcoop2.message.utils.WWWAuthenticateErrorCode;
import org.openspcoop2.message.utils.WWWAuthenticateGenerator;
import org.openspcoop2.pdd.config.DBTransazioniManager;
import org.openspcoop2.pdd.config.OpenSPCoop2Properties;
import org.openspcoop2.pdd.config.Resource;
import org.openspcoop2.pdd.config.dynamic.PddPluginLoader;
import org.openspcoop2.pdd.core.AbstractCore;
import org.openspcoop2.pdd.core.PdDContext;
import org.openspcoop2.pdd.core.autenticazione.pa.DatiInvocazionePortaApplicativa;
import org.openspcoop2.pdd.core.autenticazione.pa.EsitoAutenticazionePortaApplicativa;
import org.openspcoop2.pdd.core.autenticazione.pa.IAutenticazionePortaApplicativa;
import org.openspcoop2.pdd.core.autenticazione.pd.DatiInvocazionePortaDelegata;
import org.openspcoop2.pdd.core.autenticazione.pd.EsitoAutenticazionePortaDelegata;
import org.openspcoop2.pdd.core.autenticazione.pd.IAutenticazionePortaDelegata;
import org.openspcoop2.pdd.core.connettori.InfoConnettoreIngresso;
import org.openspcoop2.pdd.core.state.IOpenSPCoopState;
import org.openspcoop2.pdd.core.state.OpenSPCoopState;
import org.openspcoop2.pdd.core.token.Costanti;
import org.openspcoop2.pdd.core.token.InformazioniToken;
import org.openspcoop2.pdd.logger.OpenSPCoop2Logger;
import org.openspcoop2.protocol.engine.SecurityTokenUtilities;
import org.openspcoop2.protocol.sdk.Context;
import org.openspcoop2.protocol.sdk.IProtocolFactory;
import org.openspcoop2.protocol.sdk.PDNDTokenInfo;
import org.openspcoop2.protocol.sdk.SecurityToken;
import org.openspcoop2.protocol.sdk.constants.ErroriCooperazione;
import org.openspcoop2.protocol.sdk.constants.ErroriIntegrazione;
import org.openspcoop2.protocol.sdk.constants.IntegrationFunctionError;
import org.openspcoop2.protocol.sdk.state.RequestInfo;
import org.openspcoop2.utils.Utilities;
import org.openspcoop2.utils.UtilsException;
import org.openspcoop2.utils.cache.Cache;
import org.openspcoop2.utils.cache.CacheAlgorithm;
import org.openspcoop2.utils.cache.CacheType;
import org.openspcoop2.utils.cache.CacheResponse;
import org.openspcoop2.utils.date.DateManager;
import org.slf4j.Logger;

/**
 * Classe utilizzata per la gestione del processo di autenticazione Buste
 *
 * @author Poli Andrea (apoli@link.it)
 * @author $Author$
 * @version $Rev$, $Date$
 */
public class GestoreAutenticazione {

	
	/** Chiave della cache per l'autenticazione Buste  */
	private static final String AUTENTICAZIONE_CACHE_NAME = "autenticazione";
	/** Cache */
	private static Cache cacheAutenticazione = null;
	//private static final Boolean semaphoreAutenticazionePD = true;
	//private static final Boolean semaphoreAutenticazionePA = true;
	//private static final Boolean semaphoreCredenzialiMittente = true;
	
	private static final Map<String, org.openspcoop2.utils.Semaphore> _lockAutenticazionePD = new HashMap<>(); 
	private static synchronized org.openspcoop2.utils.Semaphore initLockAutenticazionePD(String tipoAutenticazione){
		org.openspcoop2.utils.Semaphore s = _lockAutenticazionePD.get(tipoAutenticazione);
		if(s==null) {
			Integer permits = OpenSPCoop2Properties.getInstance().getAutenticazioneLockPermits(tipoAutenticazione);
			if(permits==null) {
				permits = OpenSPCoop2Properties.getInstance().getAutenticazioneLockPermits();
			}
			if(permits!=null && permits.intValue()>1) {
				s = new org.openspcoop2.utils.Semaphore("GestoreAutenticazioneFruizioni_"+tipoAutenticazione, permits);
			}
			else {
				s = new org.openspcoop2.utils.Semaphore("GestoreAutenticazioneFruizioni_"+tipoAutenticazione);
			}
			_lockAutenticazionePD.put(tipoAutenticazione, s);
		}
		return s;
	}
	private static org.openspcoop2.utils.Semaphore getLockAutenticazionePD(String tipoAutenticazione){
		org.openspcoop2.utils.Semaphore s = _lockAutenticazionePD.get(tipoAutenticazione);
		if(s==null) {
			s = initLockAutenticazionePD(tipoAutenticazione);
		}
		return s;
	}
	
	
	private static final Map<String, org.openspcoop2.utils.Semaphore> _lockAutenticazionePA = new HashMap<>(); 
	private static synchronized org.openspcoop2.utils.Semaphore initLockAutenticazionePA(String tipoAutenticazione){
		org.openspcoop2.utils.Semaphore s = _lockAutenticazionePA.get(tipoAutenticazione);
		if(s==null) {
			Integer permits = OpenSPCoop2Properties.getInstance().getAutenticazioneLockPermits(tipoAutenticazione);
			if(permits==null) {
				permits = OpenSPCoop2Properties.getInstance().getAutenticazioneLockPermits();
			}
			if(permits!=null && permits.intValue()>1) {
				s = new org.openspcoop2.utils.Semaphore("GestoreAutenticazioneErogazioni_"+tipoAutenticazione, permits);
			}
			else {
				s = new org.openspcoop2.utils.Semaphore("GestoreAutenticazioneErogazioni_"+tipoAutenticazione);
			}
			_lockAutenticazionePA.put(tipoAutenticazione, s);
		}
		return s;
	}
	private static org.openspcoop2.utils.Semaphore getLockAutenticazionePA(String tipoAutenticazione){
		org.openspcoop2.utils.Semaphore s = _lockAutenticazionePA.get(tipoAutenticazione);
		if(s==null) {
			s = initLockAutenticazionePA(tipoAutenticazione);
		}
		return s;
	}
	
	private static final org.openspcoop2.utils.Semaphore lockCredenzialiMittente = new org.openspcoop2.utils.Semaphore("GestoreCredenziali");
	private static final org.openspcoop2.utils.Semaphore lockCredenzialiMittenteCreazione = new org.openspcoop2.utils.Semaphore("GestoreCredenziali-Creazione");
	private static final org.openspcoop2.utils.Semaphore lockCredenzialiMittenteAggiornamento = new org.openspcoop2.utils.Semaphore("GestoreCredenziali-Aggiornamento");
	/** Logger log */
	private static Logger logger = null;
	private static Logger logConsole = OpenSPCoop2Logger.getLoggerOpenSPCoopConsole();

	
	private static java.util.Random _rnd = null;
	private static synchronized void initRandom() {
		if(_rnd==null) {
			_rnd = new SecureRandom();
		}
	}
	public static java.util.Random getRandom() {
		if(_rnd==null) {
			initRandom();
		}
		return _rnd;
	}
	

	/* --------------- Cache --------------------*/
	public static void resetCache() throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null){
			try{
				GestoreAutenticazione.cacheAutenticazione.clear();
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}
	}
	public static String printStatsCache(String separator) throws AutenticazioneException{
		try{
			if(GestoreAutenticazione.cacheAutenticazione!=null){
				return GestoreAutenticazione.cacheAutenticazione.printStats(separator);
			}
			else{
				throw new Exception("Cache non abilitata");
			}
		}catch(Exception e){
			throw new AutenticazioneException("Visualizzazione Statistiche riguardante la cache dell'autenticazione non riuscita: "+e.getMessage(),e);
		}
	}
	public static void abilitaCache() throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null)
			throw new AutenticazioneException("Cache gia' abilitata");
		else{
			_abilitaCache();
		}
	}
	private static synchronized void _abilitaCache() throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione==null) {
			try{
				GestoreAutenticazione.cacheAutenticazione = new Cache(CacheType.JCS, GestoreAutenticazione.AUTENTICAZIONE_CACHE_NAME); // lascio JCS come default abilitato via jmx
				GestoreAutenticazione.cacheAutenticazione.build();
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}
	}
	public static void abilitaCache(Long dimensioneCache,Boolean algoritmoCacheLRU,Long itemIdleTime,Long itemLifeSecond) throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null)
			throw new AutenticazioneException("Cache gia' abilitata");
		else{
			try{
				int dimensioneCacheInt = -1;
				if(dimensioneCache!=null){
					dimensioneCacheInt = dimensioneCache.intValue();
				}
				
				String algoritmoCache = null;
				if(algoritmoCacheLRU!=null){
					if(algoritmoCacheLRU)
						 algoritmoCache = CostantiConfigurazione.CACHE_LRU.toString();
					else
						 algoritmoCache = CostantiConfigurazione.CACHE_MRU.toString();
				}else{
					algoritmoCache = CostantiConfigurazione.CACHE_LRU.toString();
				}
				
				long itemIdleTimeLong = -1;
				if(itemIdleTime!=null){
					itemIdleTimeLong = itemIdleTime;
				}
				
				long itemLifeSecondLong = -1;
				if(itemLifeSecond!=null){
					itemLifeSecondLong = itemLifeSecond;
				}
				
				GestoreAutenticazione.initCacheAutenticazione(CacheType.JCS, dimensioneCacheInt, algoritmoCache, itemIdleTimeLong, itemLifeSecondLong, null); // lascio JCS come default abilitato via jmx
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}
	}
	public static void disabilitaCache() throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione==null)
			throw new AutenticazioneException("Cache gia' disabilitata");
		else{
			_disabilitaCache();
		}
	}
	private static synchronized void _disabilitaCache() throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null) {
			try{
				GestoreAutenticazione.cacheAutenticazione.clear();
				GestoreAutenticazione.cacheAutenticazione = null;
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}
	}
	public static boolean isCacheAbilitata(){
		return GestoreAutenticazione.cacheAutenticazione != null;
	}
	public static String listKeysCache(String separator) throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null){
			try{
				return GestoreAutenticazione.cacheAutenticazione.printKeys(separator);
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}else{
			throw new AutenticazioneException("Cache non abilitata");
		}
	}
	public static List<String> listKeysCache() throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null){
			try{
				return GestoreAutenticazione.cacheAutenticazione.keys();
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}else{
			throw new AutenticazioneException("Cache non abilitata");
		}
	}
	public static String getObjectCache(String key) throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null){
			try{
				Object o = GestoreAutenticazione.cacheAutenticazione.get(key);
				if(o!=null){
					return o.toString();
				}else{
					return "oggetto con chiave ["+key+"] non presente";
				}
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}else{
			throw new AutenticazioneException("Cache non abilitata");
		}
	}
	public static Object getRawObjectCache(String key) throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null){
			try{
				Object o = GestoreAutenticazione.cacheAutenticazione.get(key);
				if(o!=null){
					if(o instanceof CacheResponse) {
						CacheResponse cR = (CacheResponse) o;
						if(cR.getObject()!=null) {
							o = cR.getObject();
						}
						else if(cR.getException()!=null) {
							o = cR.getException();
						}
					}
					return o;
				}else{
					return null;
				}
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}else{
			throw new AutenticazioneException("Cache non abilitata");
		}
	}
	public static void removeObjectCache(String key) throws AutenticazioneException{
		if(GestoreAutenticazione.cacheAutenticazione!=null){
			try{
				GestoreAutenticazione.cacheAutenticazione.remove(key);
			}catch(Exception e){
				throw new AutenticazioneException(e.getMessage(),e);
			}
		}else{
			throw new AutenticazioneException("Cache non abilitata");
		}
	}
	


	/*----------------- INIZIALIZZAZIONE --------------------*/
	public static void initialize(Logger log) throws Exception{
		GestoreAutenticazione.initialize(null, false, -1,null,-1l,-1l, log);
	}
	public static void initialize(CacheType cacheType, int dimensioneCache,String algoritmoCache,
			long idleTime, long itemLifeSecond, Logger log) throws Exception{
		GestoreAutenticazione.initialize(cacheType, true, dimensioneCache,algoritmoCache,idleTime,itemLifeSecond, log);
	}

	private static void initialize(CacheType cacheType, boolean cacheAbilitata,int dimensioneCache,String algoritmoCache,
			long idleTime, long itemLifeSecond, Logger log) throws Exception{

		// Inizializzo log
		GestoreAutenticazione.logger = log;
		
		// Inizializzazione Cache
		if(cacheAbilitata){
			GestoreAutenticazione.initCacheAutenticazione(cacheType, dimensioneCache, algoritmoCache, idleTime, itemLifeSecond, log);
		}

	}


	public static void initCacheAutenticazione(CacheType cacheType, int dimensioneCache,String algoritmoCache,
			long idleTime, long itemLifeSecond, Logger log) throws Exception {
		
		if(log!=null)
			log.info("Inizializzazione cache Autenticazione");

		GestoreAutenticazione.cacheAutenticazione = new Cache(cacheType, GestoreAutenticazione.AUTENTICAZIONE_CACHE_NAME);

		if( (dimensioneCache>0) ||
				(algoritmoCache != null) ){

			if( dimensioneCache>0 ){
				try{
					String msg = "Dimensione della cache (Autenticazione) impostata al valore: "+dimensioneCache;
					if(log!=null)
						log.info(msg);
					GestoreAutenticazione.logConsole.info(msg);
					GestoreAutenticazione.cacheAutenticazione.setCacheSize(dimensioneCache);
				}catch(Exception error){
					throw new AutenticazioneException("Parametro errato per la dimensione della cache (Gestore Messaggi): "+error.getMessage(),error);
				}
			}
			if(algoritmoCache != null ){
				String msg = "Algoritmo di cache (Autenticazione) impostato al valore: "+algoritmoCache;
				if(log!=null)
					log.info(msg);
				GestoreAutenticazione.logConsole.info(msg);
				if(CostantiConfigurazione.CACHE_MRU.toString().equalsIgnoreCase(algoritmoCache))
					GestoreAutenticazione.cacheAutenticazione.setCacheAlgoritm(CacheAlgorithm.MRU);
				else
					GestoreAutenticazione.cacheAutenticazione.setCacheAlgoritm(CacheAlgorithm.LRU);
			}

		}

		if( idleTime > 0  ){
			try{
				String msg = "Attributo 'IdleTime' (Autenticazione) impostato al valore: "+idleTime;
				if(log!=null)
					log.info(msg);
				GestoreAutenticazione.logConsole.info(msg);
				GestoreAutenticazione.cacheAutenticazione.setItemIdleTime(idleTime);
			}catch(Exception error){
				throw new AutenticazioneException("Parametro errato per l'attributo 'IdleTime' (Gestore Messaggi): "+error.getMessage(),error);
			}
		}

		try{
			String msg = "Attributo 'MaxLifeSecond' (Autenticazione) impostato al valore: "+itemLifeSecond;
			if(log!=null)
				log.info(msg);
			GestoreAutenticazione.logConsole.info(msg);
			GestoreAutenticazione.cacheAutenticazione.setItemLifeTime(itemLifeSecond);
		}catch(Exception error){
			throw new AutenticazioneException("Parametro errato per l'attributo 'MaxLifeSecond' (Gestore Messaggi): "+error.getMessage(),error);
		}

		GestoreAutenticazione.cacheAutenticazione.build();
	}
	
	@SuppressWarnings("deprecation")
	@Deprecated
	public static void disableSyncronizedGet() throws UtilsException {
		if(GestoreAutenticazione.cacheAutenticazione==null) {
			throw new UtilsException("Cache disabled");
		}
		GestoreAutenticazione.cacheAutenticazione.disableSyncronizedGet();
	}
	@SuppressWarnings("deprecation")
	@Deprecated
	public static boolean isDisableSyncronizedGet() throws UtilsException {
		if(GestoreAutenticazione.cacheAutenticazione==null) {
			throw new UtilsException("Cache disabled");
		}
		return GestoreAutenticazione.cacheAutenticazione.isDisableSyncronizedGet();
	}
	

	
	
	
	/*----------------- CLEANER --------------------*/
	
	public static void removePortaApplicativa(IDPortaApplicativa idPA) throws Exception {
		if(GestoreAutenticazione.isCacheAbilitata()) {
			List<String> keyForClean = new ArrayList<>();
			List<String> keys = GestoreAutenticazione.listKeysCache();
			if(keys!=null && !keys.isEmpty()) {
				String match = IDPortaApplicativa.PORTA_APPLICATIVA_PREFIX+idPA.getNome()+IDPortaApplicativa.PORTA_APPLICATIVA_SUFFIX;
				for (String key : keys) {
					if(key!=null && key.contains(match)) {
						keyForClean.add(key);
					}
				}
			}
			if(keyForClean!=null && !keyForClean.isEmpty()) {
				for (String key : keyForClean) {
					removeObjectCache(key);
				}
			}
		}
	}
	
	public static void removePortaDelegata(IDPortaDelegata idPD) throws Exception {
		if(GestoreAutenticazione.isCacheAbilitata()) {
			List<String> keyForClean = new ArrayList<>();
			List<String> keys = GestoreAutenticazione.listKeysCache();
			if(keys!=null && !keys.isEmpty()) {
				String match = IDPortaDelegata.PORTA_DELEGATA_PREFIX+idPD.getNome()+IDPortaDelegata.PORTA_DELEGATA_SUFFIX;
				for (String key : keys) {
					if(key!=null && key.contains(match)) {
						keyForClean.add(key);
					}
				}
			}
			if(keyForClean!=null && !keyForClean.isEmpty()) {
				for (String key : keyForClean) {
					removeObjectCache(key);
				}
			}
		}
	}
	
	public static void removeSoggetto(IDSoggetto idSoggetto) throws Exception {
		if(GestoreAutenticazione.isCacheAbilitata()) {
			List<String> keyForClean = new ArrayList<>();
			List<String> keys = GestoreAutenticazione.listKeysCache();
			if(keys!=null && !keys.isEmpty()) {
				for (String key : keys) {
					if(key!=null) {
						Object o = GestoreAutenticazione.getRawObjectCache(key);
						if(o!=null) {
							if(o instanceof EsitoAutenticazionePortaApplicativa) {
								EsitoAutenticazionePortaApplicativa esito = (EsitoAutenticazionePortaApplicativa) o;
								if(esito.getIdSoggetto()!=null && esito.getIdSoggetto().equals(idSoggetto)) {
									keyForClean.add(key);
								}
							}
						}
					}
				}
			}
			if(keyForClean!=null && !keyForClean.isEmpty()) {
				for (String key : keyForClean) {
					removeObjectCache(key);
				}
			}
		}
	}
	
	public static void removeApplicativo(IDServizioApplicativo idApplicativo) throws Exception {
		if(GestoreAutenticazione.isCacheAbilitata()) {
			List<String> keyForClean = new ArrayList<>();
			List<String> keys = GestoreAutenticazione.listKeysCache();
			if(keys!=null && !keys.isEmpty()) {
				for (String key : keys) {
					if(key!=null) {
						Object o = GestoreAutenticazione.getRawObjectCache(key);
						if(o!=null) {
							if(o instanceof EsitoAutenticazionePortaDelegata) {
								EsitoAutenticazionePortaDelegata esito = (EsitoAutenticazionePortaDelegata) o;
								if(esito.getIdServizioApplicativo()!=null && esito.getIdServizioApplicativo().equals(idApplicativo)) {
									keyForClean.add(key);
								}
							}
							else if(o instanceof EsitoAutenticazionePortaApplicativa) {
								EsitoAutenticazionePortaApplicativa esito = (EsitoAutenticazionePortaApplicativa) o;
								if(esito.getIdServizioApplicativo()!=null && esito.getIdServizioApplicativo().equals(idApplicativo)) {
									keyForClean.add(key);
								}
							}
						}
					}
				}
			}
			if(keyForClean!=null && !keyForClean.isEmpty()) {
				for (String key : keyForClean) {
					removeObjectCache(key);
				}
			}
		}
	}
	
	
	
	
	
	/*----------------- AUTENTICAZIONE --------------------*/
	
    public static EsitoAutenticazionePortaDelegata verificaAutenticazionePortaDelegata(String tipoAutenticazione, 
    		DatiInvocazionePortaDelegata datiInvocazione, ParametriAutenticazione parametriAutenticazione,
 		  PdDContext pddContext,IProtocolFactory<?> protocolFactory, OpenSPCoop2Message msg) throws Exception{
    	return _verificaAutenticazionePortaDelegata(false, tipoAutenticazione, 
    			datiInvocazione, parametriAutenticazione, 
    			pddContext, protocolFactory, msg);
    }
    public static EsitoAutenticazionePortaDelegata verificaAutenticazioneMessageBox(String tipoAutenticazione, 
    		DatiInvocazionePortaDelegata datiInvocazione, ParametriAutenticazione parametriAutenticazione,
   		  PdDContext pddContext,IProtocolFactory<?> protocolFactory) throws Exception{
    	return _verificaAutenticazionePortaDelegata(true, tipoAutenticazione, 
    			datiInvocazione, parametriAutenticazione, 
    			pddContext, protocolFactory, null);
    }
    
    private static EsitoAutenticazionePortaDelegata _verificaAutenticazionePortaDelegata(boolean messageBox, String tipoAutenticazione, 
    		DatiInvocazionePortaDelegata datiInvocazione, ParametriAutenticazione parametriAutenticazione,
   		  PdDContext pddContext,IProtocolFactory<?> protocolFactory, OpenSPCoop2Message msg) throws Exception{
    	
    	checkDatiPortaDelegata(messageBox,datiInvocazione);
    	
    	IAutenticazionePortaDelegata auth = newInstanceAuthPortaDelegata(tipoAutenticazione, pddContext, protocolFactory);
    	auth.initParametri(parametriAutenticazione);
    	if(messageBox) {
    		auth.setLogError(false);
    	}
    	
    	try {
	    	if(GestoreAutenticazione.cacheAutenticazione==null || !auth.saveAuthenticationResultInCache()){
	    		return auth.process(datiInvocazione);
			}
	    	else{
	    		String keyCache = buildCacheKey(messageBox, true, tipoAutenticazione, datiInvocazione.getKeyCache(), auth.getSuffixKeyAuthenticationResultInCache(datiInvocazione));
	
	    		// Fix: devo prima verificare se ho la chiave in cache prima di mettermi in sincronizzazione.
	    		org.openspcoop2.utils.cache.CacheResponse response = 
						(org.openspcoop2.utils.cache.CacheResponse) GestoreAutenticazione.cacheAutenticazione.get(keyCache);
				if(response != null){
					if(response.getObject()!=null){
						GestoreAutenticazione.logger.debug("Oggetto (tipo:"+response.getObject().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaDelegata) in cache.");
						EsitoAutenticazionePortaDelegata esito = (EsitoAutenticazionePortaDelegata) response.getObject();
						esito.setEsitoPresenteInCache(true);
						return esito;
					}else if(response.getException()!=null){
						GestoreAutenticazione.logger.debug("Eccezione (tipo:"+response.getException().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaDelegata) in cache.");
						throw (Exception) response.getException();
					}else{
						GestoreAutenticazione.logger.error("In cache non e' presente ne un oggetto ne un'eccezione.");
					}
				}
	    		
				String idTransazione = (pddContext!=null && pddContext.containsKey(org.openspcoop2.core.constants.Costanti.ID_TRANSAZIONE)) ? PdDContext.getValue(org.openspcoop2.core.constants.Costanti.ID_TRANSAZIONE, pddContext) : null;
				//synchronized (GestoreAutenticazione.semaphoreAutenticazionePD) {
				org.openspcoop2.utils.Semaphore lockAutenticazionePD = getLockAutenticazionePD(tipoAutenticazione);
				lockAutenticazionePD.acquire("verificaAutenticazionePortaDelegata", idTransazione);
				try {
					
					response = 
						(org.openspcoop2.utils.cache.CacheResponse) GestoreAutenticazione.cacheAutenticazione.get(keyCache);
					if(response != null){
						if(response.getObject()!=null){
							GestoreAutenticazione.logger.debug("Oggetto (tipo:"+response.getObject().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaDelegata) in cache.");
							EsitoAutenticazionePortaDelegata esito = (EsitoAutenticazionePortaDelegata) response.getObject();
							esito.setEsitoPresenteInCache(true);
							return esito;
						}else if(response.getException()!=null){
							GestoreAutenticazione.logger.debug("Eccezione (tipo:"+response.getException().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaDelegata) in cache.");
							throw (Exception) response.getException();
						}else{
							GestoreAutenticazione.logger.error("In cache non e' presente ne un oggetto ne un'eccezione.");
						}
					}
	
					// Effettuo la query
					GestoreAutenticazione.logger.debug("oggetto con chiave ["+keyCache+"] (method:verificaAutenticazionePortaDelegata) ricerco nella configurazione...");
					EsitoAutenticazionePortaDelegata esito = auth.process(datiInvocazione);
	
					// Aggiungo la risposta in cache (se esiste una cache)	
					// Sempre. Se la risposta non deve essere cachata l'implementazione può in alternativa:
					// - impostare una eccezione di processamento (che setta automaticamente noCache a true)
					// - impostare il noCache a true
					if(esito!=null){
						if(!esito.isNoCache()){
							GestoreAutenticazione.logger.info("Aggiungo oggetto ["+keyCache+"] in cache");
							try{	
								org.openspcoop2.utils.cache.CacheResponse responseCache = new org.openspcoop2.utils.cache.CacheResponse();
								responseCache.setObject(esito);
								GestoreAutenticazione.cacheAutenticazione.put(keyCache,responseCache);
							}catch(UtilsException e){
								GestoreAutenticazione.logger.error("Errore durante l'inserimento in cache ["+keyCache+"]: "+e.getMessage());
							}
						}
						return esito;
					}else{
						throw new AutenticazioneException("Metodo (GestoreAutenticazione.autenticazionePortaDelegata.process) ha ritornato un valore di esito null");
					}
				}finally{
					lockAutenticazionePD.release("verificaAutenticazionePortaDelegata", idTransazione);
				}
	    	}
    	}finally {
    		if(msg!=null) {
    			auth.cleanPostAuth(msg);
    		}
    	}
    }
	
    public static EsitoAutenticazionePortaApplicativa verificaAutenticazionePortaApplicativa(String tipoAutenticazione, 
    		DatiInvocazionePortaApplicativa datiInvocazione, ParametriAutenticazione parametriAutenticazione,
    		PdDContext pddContext,IProtocolFactory<?> protocolFactory, OpenSPCoop2Message msg) throws Exception{
    	  
    	checkDatiPortaApplicativa(datiInvocazione);
    	
    	IAutenticazionePortaApplicativa auth = newInstanceAuthPortaApplicativa(tipoAutenticazione, pddContext, protocolFactory);
    	auth.initParametri(parametriAutenticazione);
    	
    	try {
	    	if(GestoreAutenticazione.cacheAutenticazione==null || !auth.saveAuthenticationResultInCache()){
	    		return auth.process(datiInvocazione);
			}
	    	else{
	    		String keyCache = buildCacheKey(false, false, tipoAutenticazione, datiInvocazione.getKeyCache(), auth.getSuffixKeyAuthenticationResultInCache(datiInvocazione));
	
	    		// Fix: devo prima verificare se ho la chiave in cache prima di mettermi in sincronizzazione.
				org.openspcoop2.utils.cache.CacheResponse response = 
						(org.openspcoop2.utils.cache.CacheResponse) GestoreAutenticazione.cacheAutenticazione.get(keyCache);
				if(response != null){
					if(response.getObject()!=null){
						GestoreAutenticazione.logger.debug("Oggetto (tipo:"+response.getObject().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaApplicativa) in cache.");
						EsitoAutenticazionePortaApplicativa esito = (EsitoAutenticazionePortaApplicativa) response.getObject();
						esito.setEsitoPresenteInCache(true);
						return esito;
					}else if(response.getException()!=null){
						GestoreAutenticazione.logger.debug("Eccezione (tipo:"+response.getException().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaApplicativa) in cache.");
						throw (Exception) response.getException();
					}else{
						GestoreAutenticazione.logger.error("In cache non e' presente ne un oggetto ne un'eccezione.");
					}
				}
	    		
				String idTransazione = (pddContext!=null && pddContext.containsKey(org.openspcoop2.core.constants.Costanti.ID_TRANSAZIONE)) ? PdDContext.getValue(org.openspcoop2.core.constants.Costanti.ID_TRANSAZIONE, pddContext) : null;
				//synchronized (GestoreAutenticazione.semaphoreAutenticazionePA) {
				org.openspcoop2.utils.Semaphore lockAutenticazionePA = getLockAutenticazionePA(tipoAutenticazione);
				lockAutenticazionePA.acquire("verificaAutenticazionePortaApplicativa", idTransazione);
				try {
					
					response = 
						(org.openspcoop2.utils.cache.CacheResponse) GestoreAutenticazione.cacheAutenticazione.get(keyCache);
					if(response != null){
						if(response.getObject()!=null){
							GestoreAutenticazione.logger.debug("Oggetto (tipo:"+response.getObject().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaApplicativa) in cache.");
							EsitoAutenticazionePortaApplicativa esito = (EsitoAutenticazionePortaApplicativa) response.getObject();
							esito.setEsitoPresenteInCache(true);
							return esito;
						}else if(response.getException()!=null){
							GestoreAutenticazione.logger.debug("Eccezione (tipo:"+response.getException().getClass().getName()+") con chiave ["+keyCache+"] (method:verificaAutenticazionePortaApplicativa) in cache.");
							throw (Exception) response.getException();
						}else{
							GestoreAutenticazione.logger.error("In cache non e' presente ne un oggetto ne un'eccezione.");
						}
					}
	
					// Effettuo la query
					GestoreAutenticazione.logger.debug("oggetto con chiave ["+keyCache+"] (method:verificaAutenticazionePortaApplicativa) ricerco nella configurazione...");
					EsitoAutenticazionePortaApplicativa esito = auth.process(datiInvocazione);
	
					// Aggiungo la risposta in cache (se esiste una cache)	
					// Sempre. Se la risposta non deve essere cachata l'implementazione può in alternativa:
					// - impostare una eccezione di processamento (che setta automaticamente noCache a true)
					// - impostare il noCache a true
					if(esito!=null){
						if(!esito.isNoCache()){
							GestoreAutenticazione.logger.info("Aggiungo oggetto ["+keyCache+"] in cache");
							try{	
								org.openspcoop2.utils.cache.CacheResponse responseCache = new org.openspcoop2.utils.cache.CacheResponse();
								responseCache.setObject(esito);
								GestoreAutenticazione.cacheAutenticazione.put(keyCache,responseCache);
							}catch(UtilsException e){
								GestoreAutenticazione.logger.error("Errore durante l'inserimento in cache ["+keyCache+"]: "+e.getMessage());
							}
						}
						return esito;
					}else{
						throw new AutenticazioneException("Metodo (GestoreAutenticazione.autenticazionePortaApplicativa.process) ha ritornato un valore di esito null");
					}
				}finally {
					lockAutenticazionePA.release("verificaAutenticazionePortaApplicativa", idTransazione);
				}
	    	}
    	}finally {
    		if(msg!=null) {
    			auth.cleanPostAuth(msg);
    		}
    	}
    }
        
    private static void checkDatiPortaDelegata(boolean messageBox, DatiInvocazionePortaDelegata datiInvocazione) throws AutenticazioneException{
    	
    	InfoConnettoreIngresso infoConnettoreIngresso = datiInvocazione.getInfoConnettoreIngresso();
    	if(infoConnettoreIngresso==null)
			throw new AutenticazioneException("(Parametri) InfoConnettoreIngresso non definito");
		if(infoConnettoreIngresso.getCredenziali()==null)
			throw new AutenticazioneException("(Parametri) InfoConnettoreIngresso.credenziali non definito");
		
		if(messageBox==false){
	    	IDPortaDelegata idPD = datiInvocazione.getIdPD();
	    	if(idPD==null)
				throw new AutenticazioneException("(Parametri) IDPortaDelegata non definito");
	    	if(idPD.getNome()==null)
				throw new AutenticazioneException("(Parametri) IDPortaDelegata.nome non definito");
			
			PortaDelegata pd = datiInvocazione.getPd();
			if(pd==null)
				throw new AutenticazioneException("(Parametri) PortaDelegata non definito");
		}
		
    }
    
    private static void checkDatiPortaApplicativa(DatiInvocazionePortaApplicativa datiInvocazione) throws AutenticazioneException{
    	
    	InfoConnettoreIngresso infoConnettoreIngresso = datiInvocazione.getInfoConnettoreIngresso();
    	if(infoConnettoreIngresso==null)
			throw new AutenticazioneException("(Parametri) InfoConnettoreIngresso non definito");
		if(infoConnettoreIngresso.getCredenziali()==null)
			throw new AutenticazioneException("(Parametri) InfoConnettoreIngresso.credenziali non definito");

		// Dipende dal ruolo della Busta, se c'e' la porta applicativa o delegata
//    	IDPortaApplicativaByNome idPA = datiInvocazione.getIdPA();
//    	if(idPA==null)
//			throw new AutenticazioneException("(Parametri) IDPortaApplicativaByNome non definito");
//    	if(idPA.getNome()==null)
//			throw new AutenticazioneException("(Parametri) IDPortaApplicativaByNome.nome non definito");
//    	IDSoggetto idSoggettoProprietarioPA = idPA.getSoggetto();
//		if(idSoggettoProprietarioPA==null || idSoggettoProprietarioPA.getTipo()==null || idSoggettoProprietarioPA.getNome()==null)
//			throw new AutenticazioneException("(Parametri) IDPortaApplicativaByNome.Soggetto non definito");
//		
//		PortaApplicativa pa = datiInvocazione.getPa();
//		if(pa==null)
//			throw new AutenticazioneException("(Parametri) PortaApplicativa non definito");
		
    }
    
    private static String buildCacheKey(boolean messageBox, boolean portaDelegata, String tipoAutenticazione, String keyCache, String suffixKeyCache) throws AutenticazioneException{
    	StringBuilder bf = new StringBuilder();
    	
    	if(portaDelegata){
    		if(messageBox){
    			bf.append("MessageBox ");
    		}
    		else{
    			bf.append("PD ");
    		}
    	}else{
    		bf.append("PA ");
    	}
    	
    	bf.append(" Auth:").append(tipoAutenticazione).append(" ");
    	
    	bf.append(keyCache);
    	
    	if(suffixKeyCache!=null && !"".equals(suffixKeyCache)){
    		bf.append(" ");
    		bf.append(suffixKeyCache);
    	}
    	
    	return bf.toString();
    }
    
    private static IAutenticazionePortaDelegata newInstanceAuthPortaDelegata(String tipoAutenticazione,PdDContext pddContext, IProtocolFactory<?> protocolFactory) throws AutenticazioneException{
    	IAutenticazionePortaDelegata auth = null;
		try{
			PddPluginLoader pluginLoader = PddPluginLoader.getInstance();
			auth = (IAutenticazionePortaDelegata) pluginLoader.newAutenticazionePortaDelegata(tipoAutenticazione);
		}catch(Exception e){
			throw new AutenticazioneException(e.getMessage(),e); // descrizione errore già corretta
		}
		String classType = null; 
		try{
			classType = auth.getClass().getName();
			AbstractCore.init(auth, pddContext, protocolFactory);
			return auth;
		}catch(Exception e){
			throw new AutenticazioneException("Riscontrato errore durante l'inizializzazione della classe ["+classType+
					"] che definisce l'autenticazione della fruizione: "+e.getMessage(),e);
		}
    }
    
    private static IAutenticazionePortaApplicativa newInstanceAuthPortaApplicativa(String tipoAutenticazione,PdDContext pddContext, IProtocolFactory<?> protocolFactory) throws AutenticazioneException{
    	IAutenticazionePortaApplicativa auth = null;
		try{
			PddPluginLoader pluginLoader = PddPluginLoader.getInstance();
			auth = (IAutenticazionePortaApplicativa) pluginLoader.newAutenticazionePortaApplicativa(tipoAutenticazione);
		}catch(Exception e){
			throw new AutenticazioneException(e.getMessage(),e); // descrizione errore già corretta
		}
		String classType = null; 
    	try{
			classType = auth.getClass().getName();
			AbstractCore.init(auth, pddContext, protocolFactory);
			return auth;
		}catch(Exception e){
			throw new AutenticazioneException("Riscontrato errore durante l'inizializzazione della classe ["+classType+
					"] che definisce l'autenticazione della erogazione: "+e.getMessage(),e);
		}
    }
    
    public static boolean isAutenticazioneTokenEnabled(GestioneTokenAutenticazione gestioneTokenAutenticazione) {
    	return
				StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getIssuer()) ||
				StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getClientId()) ||
				StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getSubject()) ||
				StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getUsername()) ||
				StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getEmail());
    }
    
    public static String getLabel(GestioneTokenAutenticazione gestioneTokenAutenticazione) {
    	return _getActions(gestioneTokenAutenticazione);
    }
    public static String getActions(GestioneTokenAutenticazione gestioneTokenAutenticazione) {
    	return _getActions(gestioneTokenAutenticazione);
    }
    private static String _getActions(GestioneTokenAutenticazione gestioneTokenAutenticazione) {
    	StringBuilder bf = new StringBuilder();
    	if(StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getIssuer())) {
    		if(bf.length()>0) {
    			bf.append(",");
    		}
    		bf.append(Costanti.GESTIONE_TOKEN_AUTENTICAZIONE_ISSUER);
    	}
    	if(StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getClientId())) {
    		if(bf.length()>0) {
    			bf.append(",");
    		}
    		bf.append(Costanti.GESTIONE_TOKEN_AUTENTICAZIONE_CLIENT_ID);
    	}
    	if(StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getSubject())) {
    		if(bf.length()>0) {
    			bf.append(",");
    		}
    		bf.append(Costanti.GESTIONE_TOKEN_AUTENTICAZIONE_SUBJECT);
    	}
    	if(StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getUsername())) {
    		if(bf.length()>0) {
    			bf.append(",");
    		}
    		bf.append(Costanti.GESTIONE_TOKEN_AUTENTICAZIONE_USERNAME);
    	}
    	if(StatoFunzionalita.ABILITATO.equals(gestioneTokenAutenticazione.getEmail())) {
    		if(bf.length()>0) {
    			bf.append(",");
    		}
    		bf.append(Costanti.GESTIONE_TOKEN_AUTENTICAZIONE_EMAIL);
    	}
    	return bf.toString();
    }
    
    public static EsitoAutenticazionePortaDelegata verificaAutenticazioneTokenPortaDelegata(GestioneTokenAutenticazione tipoAutenticazione, DatiInvocazionePortaDelegata datiInvocazione,
    		PdDContext pddContext,IProtocolFactory<?> protocolFactory, OpenSPCoop2Message msg) throws Exception{
    	
    	EsitoAutenticazionePortaDelegata esito = new EsitoAutenticazionePortaDelegata();
      	esito.setClientAuthenticated(true);
      	
      	esito = (EsitoAutenticazionePortaDelegata) autenticazioneToken(tipoAutenticazione, esito, pddContext, datiInvocazione);
      	if(esito.isClientAuthenticated()==false) {
      		esito.setErroreIntegrazione(IntegrationFunctionError.TOKEN_REQUIRED_CLAIMS_NOT_FOUND, ErroriIntegrazione.ERRORE_445_TOKEN_AUTORIZZAZIONE_FALLITA.getErroreIntegrazione());
    	}
      	return esito;
    }
    
    public static EsitoAutenticazionePortaApplicativa verificaAutenticazioneTokenPortaApplicativa(GestioneTokenAutenticazione tipoAutenticazione, DatiInvocazionePortaApplicativa datiInvocazione,
    		PdDContext pddContext,IProtocolFactory<?> protocolFactory, OpenSPCoop2Message msg) throws Exception{
    	
    	EsitoAutenticazionePortaApplicativa esito = new EsitoAutenticazionePortaApplicativa();
      	esito.setClientAuthenticated(true);
      	
      	esito = (EsitoAutenticazionePortaApplicativa) autenticazioneToken(tipoAutenticazione, esito, pddContext, datiInvocazione);
      	if(esito.isClientAuthenticated()==false) {
      		esito.setErroreCooperazione(IntegrationFunctionError.TOKEN_REQUIRED_CLAIMS_NOT_FOUND, ErroriCooperazione.TOKEN_AUTORIZZAZIONE_FALLITA.getErroreCooperazione());
    	}
      	return esito;
    	
    }
    
    private static EsitoAutenticazione autenticazioneToken(GestioneTokenAutenticazione tipoAutenticazione, EsitoAutenticazione esito, 
    		PdDContext pddContext, AbstractDatiInvocazione datiInvocazione) throws Exception {
    	
    	boolean autenticato = true;
		String errorMessage = null;
    	if(tipoAutenticazione!=null) {
    		
    		boolean issuer = StatoFunzionalita.ABILITATO.equals(tipoAutenticazione.getIssuer());
    		boolean clientId = StatoFunzionalita.ABILITATO.equals(tipoAutenticazione.getClientId());
    		boolean subject = StatoFunzionalita.ABILITATO.equals(tipoAutenticazione.getSubject());
    		boolean username = StatoFunzionalita.ABILITATO.equals(tipoAutenticazione.getUsername());
    		boolean email = StatoFunzionalita.ABILITATO.equals(tipoAutenticazione.getEmail());
    		
    		if(issuer || clientId || subject || username ||email ) {
    				
	    		InformazioniToken informazioniTokenNormalizzate = null;
	    		Object oInformazioniTokenNormalizzate = pddContext.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_INFORMAZIONI_NORMALIZZATE);
	    		if(oInformazioniTokenNormalizzate!=null) {
	    			informazioniTokenNormalizzate = (InformazioniToken) oInformazioniTokenNormalizzate;
	    		}
	    	
	    		if(issuer) {
	    			if(informazioniTokenNormalizzate==null || informazioniTokenNormalizzate.getIss()==null || "".equals(informazioniTokenNormalizzate.getIss())) {
	    				autenticato = false;
		    			errorMessage = "Token without issuer claim";
	    			}
	    		}
	    		
	    		if(clientId) {
	    			if(informazioniTokenNormalizzate==null || informazioniTokenNormalizzate.getClientId()==null || "".equals(informazioniTokenNormalizzate.getClientId())) {
	    				autenticato = false;
		    			errorMessage = "Token without clientId claim";
	    			}
	    		}
	    		
	    		if(subject) {
	    			if(informazioniTokenNormalizzate==null || informazioniTokenNormalizzate.getSub()==null || "".equals(informazioniTokenNormalizzate.getSub())) {
	    				autenticato = false;
		    			errorMessage = "Token without subject claim";
	    			}
	    		}
	    		
	    		if(username) {
	    			if(informazioniTokenNormalizzate==null || informazioniTokenNormalizzate.getUsername()==null || "".equals(informazioniTokenNormalizzate.getUsername())) {
	    				autenticato = false;
		    			errorMessage = "Token without username claim";
	    			}
	    		}
	    		
	    		if(email) {
	    			if(informazioniTokenNormalizzate==null || informazioniTokenNormalizzate.getUserInfo()==null || 
	    					informazioniTokenNormalizzate.getUserInfo().getEMail()==null || 
	    					"".equals(informazioniTokenNormalizzate.getUserInfo().getEMail())) {
	    				autenticato = false;
		    			errorMessage = "Token without email claim";
	    			}
	    		}
			}
    		
    	}
    	
    	if(!autenticato) {
    		
    		esito.setDetails(errorMessage);
    		
    		String realm = "OpenSPCoop";
			Object oRealm = pddContext.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_REALM);
    		if(oRealm!=null) {
    			realm = (String) oRealm;
    		}
			
			boolean emptyMessage = false;
			Object oEmptyMessage = pddContext.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_MESSAGE_ERROR_BODY_EMPTY);
    		if(oEmptyMessage!=null) {
    			emptyMessage = (Boolean) oEmptyMessage;
    		}
    		
    		boolean genericMessage = false;
			Object oGenericMessage = pddContext.getObject(org.openspcoop2.pdd.core.token.Costanti.PDD_CONTEXT_TOKEN_MESSAGE_ERROR_GENERIC_MESSAGE);
    		if(oGenericMessage!=null) {
    			genericMessage = (Boolean) oGenericMessage;
    		}
			
    		esito.setClientAuthenticated(false);
    		if(emptyMessage) {
    			esito.setErrorMessage(WWWAuthenticateGenerator.buildErrorMessage(WWWAuthenticateErrorCode.insufficient_scope, realm, genericMessage, errorMessage));
    		}
    		else {
    			esito.setWwwAuthenticateErrorHeader(WWWAuthenticateGenerator.buildHeaderValue(WWWAuthenticateErrorCode.insufficient_scope, realm, genericMessage, errorMessage));
    		}
		}
		
    	
    	return esito;
    }
    
    
    public static void updateCredenzialiTrasporto(IDSoggetto dominio, String modulo, String idTransazione, 
    		String tipoAutenticazione, String credential, CredenzialiMittente credenzialiMittente,
    		IOpenSPCoopState openspcoopState, String identitaChiamante, RequestInfo requestInfo) throws Exception{
    	
    	CredenzialeSearchTrasporto trasportoSearch = new CredenzialeSearchTrasporto(tipoAutenticazione);
    	trasportoSearch.disableConvertToDBValue();
		CredenzialeTrasporto trasporto = new CredenzialeTrasporto(tipoAutenticazione, credential);
    	credenzialiMittente.setTrasporto(getCredenzialeMittente(dominio, modulo, idTransazione, 
    			trasportoSearch, trasporto, openspcoopState, identitaChiamante,
				null, requestInfo));
    	
    }
    
    public static void updateCredenzialiToken(IDSoggetto dominio, String modulo, String idTransazione, 
    		InformazioniToken informazioniTokenNormalizzate, IDServizioApplicativo idApplicativoToken, CredenzialiMittente credenzialiMittente, 
    		IOpenSPCoopState openspcoopState, String identitaChiamante, RequestInfo requestInfo,
    		Context context) throws Exception{
    	
    	if(informazioniTokenNormalizzate.getIss()!=null) {
    		CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.TOKEN_ISSUER);
    		tokenSearch.disableConvertToDBValue();
    		CredenzialeToken token = new CredenzialeToken(TipoCredenzialeMittente.TOKEN_ISSUER, informazioniTokenNormalizzate.getIss());
    		credenzialiMittente.setTokenIssuer(getCredenzialeMittente(dominio, modulo, idTransazione, 
    				tokenSearch, token, openspcoopState, identitaChiamante,
    				null, requestInfo));
    	}
    	if(informazioniTokenNormalizzate.getClientId()!=null) {
    		if(idApplicativoToken==null) {
	    		CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.TOKEN_CLIENT_ID);
	    		tokenSearch.disableConvertToDBValue();
	    		CredenzialeToken token = new CredenzialeToken(TipoCredenzialeMittente.TOKEN_CLIENT_ID, informazioniTokenNormalizzate.getClientId());
	    		credenzialiMittente.setTokenClientId(getCredenzialeMittente(dominio, modulo, idTransazione, 
	    				tokenSearch, token, openspcoopState, identitaChiamante,
	    				null, requestInfo));
    		}
    		else {
    			CredenzialeSearchTokenClient tokenSearch = new CredenzialeSearchTokenClient(true, true, true);
    			tokenSearch.disableConvertToDBValue();
    			CredenzialeTokenClient token = new CredenzialeTokenClient(informazioniTokenNormalizzate.getClientId(), idApplicativoToken);
    			credenzialiMittente.setTokenClientId(getCredenzialeMittente(dominio, modulo, idTransazione, 
	    				tokenSearch, token, openspcoopState, identitaChiamante,
	    				null, requestInfo));
    		}
    	}

    	if(informazioniTokenNormalizzate.getSub()!=null) {
    		CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.TOKEN_SUBJECT);
    		tokenSearch.disableConvertToDBValue();
    		CredenzialeToken token = new CredenzialeToken(TipoCredenzialeMittente.TOKEN_SUBJECT, informazioniTokenNormalizzate.getSub());
    		credenzialiMittente.setTokenSubject(getCredenzialeMittente(dominio, modulo, idTransazione, 
    				tokenSearch, token, openspcoopState, identitaChiamante,
    				null, requestInfo));
    	}
    	if(informazioniTokenNormalizzate.getUsername()!=null) {
    		CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.TOKEN_USERNAME);
    		tokenSearch.disableConvertToDBValue();
    		CredenzialeToken token = new CredenzialeToken(TipoCredenzialeMittente.TOKEN_USERNAME, informazioniTokenNormalizzate.getUsername());
    		credenzialiMittente.setTokenUsername(getCredenzialeMittente(dominio, modulo, idTransazione, 
    				tokenSearch, token, openspcoopState, identitaChiamante,
    				null, requestInfo));
    	}
    	if(informazioniTokenNormalizzate.getUserInfo()!=null && informazioniTokenNormalizzate.getUserInfo().getEMail()!=null) {
    		CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.TOKEN_EMAIL);
    		tokenSearch.disableConvertToDBValue();
    		CredenzialeToken token = new CredenzialeToken(TipoCredenzialeMittente.TOKEN_EMAIL, informazioniTokenNormalizzate.getUserInfo().getEMail());
    		credenzialiMittente.setTokenEMail(getCredenzialeMittente(dominio, modulo, idTransazione, 
    				tokenSearch, token, openspcoopState, identitaChiamante,
    				null, requestInfo));
    	}
    	
    	updateCredenzialiTokenPDND(dominio, modulo, idTransazione, 
        		informazioniTokenNormalizzate, credenzialiMittente, 
        		openspcoopState, identitaChiamante, requestInfo,
        		context);
    	
    }
    
    public static void updateCredenzialiTokenPDND(IDSoggetto dominio, String modulo, String idTransazione, 
    		InformazioniToken informazioniTokenNormalizzate, CredenzialiMittente credenzialiMittente, 
    		IOpenSPCoopState openspcoopState, String identitaChiamante, RequestInfo requestInfo,
    		Context context) throws Exception{
    	    	
    	SecurityToken securityTokenForContext = SecurityTokenUtilities.readSecurityToken(context);
    	
    	Long refClientIdCredenzialieMittente = null;
    	if(credenzialiMittente!=null && credenzialiMittente.getTokenClientId()!=null && credenzialiMittente.getTokenClientId().getId()!=null &&
    			credenzialiMittente.getTokenClientId().getId()>0) {
    		refClientIdCredenzialieMittente = credenzialiMittente.getTokenClientId().getId();
    	}
    	if(refClientIdCredenzialieMittente==null) {
    		return;
    	}
    	
    	if(
    			(securityTokenForContext!=null && securityTokenForContext.getPdnd()!=null)
    			||
    			(informazioniTokenNormalizzate.getPdnd()!=null && !informazioniTokenNormalizzate.getPdnd().isEmpty())
    		) {
    		OpenSPCoop2Properties properties = OpenSPCoop2Properties.getInstance();
    		    		
    		if(credenzialiMittente.getTokenPdndClientJson()==null &&
    				properties.isGestoreChiaviPDNDclientsTraceJsonResponse() &&
    				(securityTokenForContext!=null && securityTokenForContext.getPdnd()!=null)) {
    			String clientJson = securityTokenForContext.getPdnd().getClientJson();
    			if(clientJson!=null && StringUtils.isNotEmpty(clientJson)) {
    				CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.PDND_CLIENT_JSON);
    	    		tokenSearch.disableConvertToDBValue();
    	    		CredenzialeTokenPDND token = new CredenzialeTokenPDND(TipoCredenzialeMittente.PDND_CLIENT_JSON, clientJson, refClientIdCredenzialieMittente);
    	    		credenzialiMittente.setTokenPdndClientJson(getCredenzialeMittente(dominio, modulo, idTransazione, 
    	    				tokenSearch, token, openspcoopState, identitaChiamante,
    	    				null, requestInfo));
    	    		/**System.out.println("["+idTransazione+"] PDND_CLIENT_JSON ref["+refClientIdCredenzialieMittente+"] ("+clientJson+") ID CREDENZIALE: "+credenzialiMittente.getTokenPdndClientJson().getId()+
    	    				" REFDB: "+credenzialiMittente.getTokenPdndClientJson().getRefCredenziale());*/
    			}
    		}
    			
    		if(credenzialiMittente.getTokenPdndOrganizationJson()==null &&
    				properties.isGestoreChiaviPDNDorganizationsTraceJsonResponse() &&
    				(securityTokenForContext!=null && securityTokenForContext.getPdnd()!=null)) {
    			String organizationJson = securityTokenForContext.getPdnd().getOrganizationJson();
    			if(organizationJson!=null && StringUtils.isNotEmpty(organizationJson)) {
    				CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.PDND_ORGANIZATION_JSON);
    	    		tokenSearch.disableConvertToDBValue();
    	    		CredenzialeTokenPDND token = new CredenzialeTokenPDND(TipoCredenzialeMittente.PDND_ORGANIZATION_JSON, organizationJson, refClientIdCredenzialieMittente);
    	    		credenzialiMittente.setTokenPdndOrganizationJson(getCredenzialeMittente(dominio, modulo, idTransazione, 
    	    				tokenSearch, token, openspcoopState, identitaChiamante,
    	    				null, requestInfo));
    	    		/**System.out.println("["+idTransazione+"] PDND_ORGANIZATION_JSON ref["+refClientIdCredenzialieMittente+"] ("+organizationJson+") ID CREDENZIALE: "+credenzialiMittente.getTokenPdndOrganizationJson().getId()+
    	    				" REFDB: "+credenzialiMittente.getTokenPdndOrganizationJson().getRefCredenziale());*/
    			}
    		}
    		
    		if(credenzialiMittente.getTokenPdndOrganizationName()==null &&
    				properties.isGestoreChiaviPDNDorganizationsTraceName()) {
    			
    			String organizationName = null;
    			if(securityTokenForContext!=null && securityTokenForContext.getPdnd()!=null){
    				try {
    					organizationName = securityTokenForContext.getPdnd().getOrganizationName(logger);
    				}catch(Exception e) {
    					// ignore (nel caso non è presente un organizationJson non è possibile recuperarlo)
    				}
    			}
    			if(organizationName==null && informazioniTokenNormalizzate.getPdnd()!=null) {
    				organizationName = PDNDTokenInfo.readOrganizationNameFromPDNDMap(informazioniTokenNormalizzate.getPdnd());
    			}
    			if(organizationName!=null && StringUtils.isNotEmpty(organizationName)) {
    				CredenzialeSearchToken tokenSearch = new CredenzialeSearchToken(TipoCredenzialeMittente.PDND_ORGANIZATION_NAME);
    	    		tokenSearch.disableConvertToDBValue();
    	    		CredenzialeTokenPDND token = new CredenzialeTokenPDND(TipoCredenzialeMittente.PDND_ORGANIZATION_NAME, organizationName, refClientIdCredenzialieMittente);
    	    		credenzialiMittente.setTokenPdndOrganizationName(getCredenzialeMittente(dominio, modulo, idTransazione, 
    	    				tokenSearch, token, openspcoopState, identitaChiamante,
    	    				null, requestInfo));
    	    		/**System.out.println("["+idTransazione+"] PDND_ORGANIZATION_NAME ref["+refClientIdCredenzialieMittente+"] ("+organizationName+") ID CREDENZIALE: "+credenzialiMittente.getTokenPdndOrganizationName().getId()+
    	    				" REFDB: "+credenzialiMittente.getTokenPdndOrganizationName().getRefCredenziale());*/
    			}
    		}
    	}
    	
    }
    
    public static CredenzialeMittente convertGruppiToCredenzialiMittenti(IDSoggetto dominio, String modulo, String idTransazione,
    		List<String> gruppiList, IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict, RequestInfo requestInfo) throws Exception{
    	if(gruppiList!=null && !gruppiList.isEmpty()) {
	    	CredenzialeSearchGruppo gruppiSearch = new CredenzialeSearchGruppo();
	    	gruppiSearch.disableConvertToDBValue();
	    	CredenzialeGruppi gruppi = new CredenzialeGruppi(gruppiList);
	    	return getCredenzialeMittente(dominio, modulo, idTransazione, 
	    			gruppiSearch, gruppi, openspcoopState, identitaChiamante,
	    			sbConflict, requestInfo);
    	}
    	return null;
    }
    
    public static CredenzialeMittente convertAPIToCredenzialiMittenti(IDSoggetto dominio, String modulo, String idTransazione,
    		String uriAccordoServizio, IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict, RequestInfo requestInfo) throws Exception{
    	if(uriAccordoServizio!=null && !"".equals(uriAccordoServizio)) {
	    	CredenzialeSearchApi apiSearch = new CredenzialeSearchApi();
	    	apiSearch.disableConvertToDBValue();
	    	CredenzialeApi api = new CredenzialeApi(uriAccordoServizio);
	    	return getCredenzialeMittente(dominio, modulo, idTransazione, 
	    			apiSearch, api, openspcoopState, identitaChiamante,
	    			sbConflict, requestInfo);
    	}
    	return null;
    }
    
    public static CredenzialeMittente convertEventiToCredenzialiMittenti(IDSoggetto dominio, String modulo, String idTransazione,
    		List<String> eventiList, IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict, RequestInfo requestInfo) throws Exception{
    	if(eventiList!=null && !eventiList.isEmpty()) {
	    	CredenzialeSearchEvento eventiSearch = new CredenzialeSearchEvento();
	    	eventiSearch.disableConvertToDBValue();
	    	CredenzialeEventi eventi = new CredenzialeEventi(eventiList);
	    	return getCredenzialeMittente(dominio, modulo, idTransazione, 
	    			eventiSearch, eventi, openspcoopState, identitaChiamante,
	    			sbConflict, requestInfo);
    	}
    	return null;
    }
    
    public static CredenzialeMittente convertClientCredentialToCredenzialiMittenti(IDSoggetto dominio, String modulo, String idTransazione,
    		String socketAddress, String transportAddress, 
    		IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict, RequestInfo requestInfo) throws Exception{
    	boolean socketAddressDefined = socketAddress!=null && !"".equals(socketAddress);
		boolean transportAddressDefined = transportAddress!=null && !"".equals(transportAddress);
    	if(socketAddressDefined || transportAddressDefined) {
	    	CredenzialeSearchClientAddress clientAddressSearch = new CredenzialeSearchClientAddress(socketAddressDefined, transportAddressDefined, true);
	    	clientAddressSearch.disableConvertToDBValue();
	    	CredenzialeClientAddress clientAddress = new CredenzialeClientAddress(socketAddress, transportAddress);
	    	return getCredenzialeMittente(dominio, modulo, idTransazione, 
	    			clientAddressSearch, clientAddress, openspcoopState, identitaChiamante,
	    			sbConflict, requestInfo);
    	}
    	return null;
    }
    
    private static CredenzialeMittente getCredenzialeMittente(IDSoggetto dominio, String modulo, String idTransazione, 
    		AbstractSearchCredenziale searchCredential, AbstractCredenziale credentialParam,
    		IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict, RequestInfo requestInfo) throws Exception{
      	
    	if(dominio==null)
			throw new AutenticazioneException("(Parametri) dominio non definito");
    	if(modulo==null)
			throw new AutenticazioneException("(Parametri) modulo non definito");
    	if(idTransazione==null)
			throw new AutenticazioneException("(Parametri) idTransazione non definito");
    	if(searchCredential==null)
			throw new AutenticazioneException("(Parametri) searchCredential non definito");
		if(credentialParam==null)
			throw new AutenticazioneException("(Parametri) credenziali non definite");
		      	
		String credential = credentialParam.getCredenziale();
		int maxLengthCredenziali = OpenSPCoop2Properties.getInstance().getTransazioniCredenzialiMittenteMaxLength();
		if(credential.length()>maxLengthCredenziali) {
			logger.error("Attenzione: credenziale '"+searchCredential.getTipo()+"' ricevuta supera la dimensione massima consentita '"+maxLengthCredenziali+"'. Verrà salvata troncata a tale dimensione. Credenziale: '"+credential+"'");
			credential = credential.substring(0,maxLengthCredenziali);
			try {
				credentialParam.updateCredenziale(credential);
			}catch(Exception e) {
				logger.error("Non è possibile troncare la credenziale di tipo '"+searchCredential.getTipo()+"', l'informazione non verrà salvata");
			}
		}
		
		boolean useRequestInfo = requestInfo!=null && requestInfo.getRequestConfig()!=null;
		String keyCache = buildCacheKey(searchCredential, credential);
		
    	int maxLifeSeconds = OpenSPCoop2Properties.getInstance().getTransazioniCredenzialiMittenteLifeSeconds();
		Date scadenzaEntry = new Date(DateManager.getTimeMillis()-(maxLifeSeconds*1000));
		
		if(useRequestInfo) {
			CredenzialeMittente c = requestInfo.getRequestConfig().getCredenzialeMittente(searchCredential.getTipoCredenzialeMittente(), keyCache);
			if(c!=null && c.getOraRegistrazione().after(scadenzaEntry)) {
				return c; // informazione in cache valida
			}
		}
		
		CredenzialeMittente c = _getCredenzialeMittente(dominio, modulo, idTransazione, 
	    		searchCredential, credentialParam,
	    		openspcoopState, identitaChiamante,
	    		sbConflict, requestInfo,
	    		keyCache, scadenzaEntry);
		
		if(useRequestInfo) {
			requestInfo.getRequestConfig().addCredenzialeMittente(searchCredential.getTipoCredenzialeMittente(), keyCache, c, idTransazione, scadenzaEntry);
		}
		
		return c;
    }
    
    private static CredenzialeMittente _getCredenzialeMittente(IDSoggetto dominio, String modulo, String idTransazione, 
    		AbstractSearchCredenziale searchCredential, AbstractCredenziale credentialParam,
    		IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict, RequestInfo requestInfo,
    		String keyCache, Date scadenzaEntry) throws Exception{
		    	
    	if(GestoreAutenticazione.cacheAutenticazione==null){
    		return _getCredenzialeMittente(dominio, modulo, idTransazione, scadenzaEntry, searchCredential, credentialParam, openspcoopState, identitaChiamante, sbConflict);
		}
    	else{
    		// Fix: devo prima verificare se ho la chiave in cache prima di mettermi in sincronizzazione.
			org.openspcoop2.utils.cache.CacheResponse response = 
					(org.openspcoop2.utils.cache.CacheResponse) GestoreAutenticazione.cacheAutenticazione.get(keyCache);
			if(response != null){
				if(response.getObject()!=null){
					GestoreAutenticazione.logger.debug("Oggetto (tipo:"+response.getObject().getClass().getName()+") con chiave ["+keyCache+"] (method:getCredenzialeMittente) in cache.");
					CredenzialeMittente credenziale = (CredenzialeMittente) response.getObject();
					if(credenziale.getOraRegistrazione().after(scadenzaEntry)) {
						return credenziale; // informazione in cache valida
					}
				}else if(response.getException()!=null){
					GestoreAutenticazione.logger.debug("Eccezione (tipo:"+response.getException().getClass().getName()+") con chiave ["+keyCache+"] (method:getCredenzialeMittente) in cache.");
					throw (Exception) response.getException();
				}else{
					GestoreAutenticazione.logger.error("In cache non e' presente ne un oggetto ne un'eccezione.");
				}
			}
    		
			//synchronized (GestoreAutenticazione.semaphoreCredenzialiMittente) {
			GestoreAutenticazione.lockCredenzialiMittente.acquire("getCredenzialeMittente", idTransazione);
			try {
				response = 
					(org.openspcoop2.utils.cache.CacheResponse) GestoreAutenticazione.cacheAutenticazione.get(keyCache);
				if(response != null){
					if(response.getObject()!=null){
						GestoreAutenticazione.logger.debug("Oggetto (tipo:"+response.getObject().getClass().getName()+") con chiave ["+keyCache+"] (method:getCredenzialeMittente) in cache.");
						CredenzialeMittente credenziale = (CredenzialeMittente) response.getObject();
						if(credenziale.getOraRegistrazione().after(scadenzaEntry)) {
							return credenziale; // informazione in cache valida
						}
					}else if(response.getException()!=null){
						GestoreAutenticazione.logger.debug("Eccezione (tipo:"+response.getException().getClass().getName()+") con chiave ["+keyCache+"] (method:getCredenzialeMittente) in cache.");
						throw (Exception) response.getException();
					}else{
						GestoreAutenticazione.logger.error("In cache non e' presente ne un oggetto ne un'eccezione.");
					}
				}

				// Effettuo la query
				GestoreAutenticazione.logger.debug("oggetto con chiave ["+keyCache+"] (method:getCredenzialeMittente) ricerco nella configurazione...");
				CredenzialeMittente credenziale = _getCredenzialeMittente(dominio, modulo, idTransazione, scadenzaEntry, searchCredential, credentialParam, openspcoopState, identitaChiamante, sbConflict);

				// Aggiungo la risposta in cache (se esiste una cache)	
				// Sempre. Se la risposta non deve essere cachata l'implementazione può in alternativa:
				// - impostare una eccezione di processamento (che setta automaticamente noCache a true)
				// - impostare il noCache a true
				if(credenziale!=null){
					GestoreAutenticazione.logger.info("Aggiungo oggetto ["+keyCache+"] in cache");
					try{
						org.openspcoop2.utils.cache.CacheResponse responseCache = new org.openspcoop2.utils.cache.CacheResponse();
						responseCache.setObject(credenziale);
						GestoreAutenticazione.cacheAutenticazione.put(keyCache,responseCache);
					}catch(UtilsException e){
						GestoreAutenticazione.logger.error("Errore durante l'inserimento in cache ["+keyCache+"]: "+e.getMessage());
					}
					return credenziale;
				}else{
					throw new AutenticazioneException("Metodo (GestoreAutenticazione.getCredenzialeMittente) ha ritornato un valore di esito null");
				}
			}finally {
				GestoreAutenticazione.lockCredenzialiMittente.release("getCredenzialeMittente", idTransazione);
			}
    	}
    }
    
    private static String buildCacheKey(AbstractSearchCredenziale searchCredential, String credential) throws AutenticazioneException{
    	StringBuilder bf = new StringBuilder();

    	bf.append(searchCredential.getTipo()).append(" ");
    	
    	bf.append(credential);
    	
    	return bf.toString();
    }
    
    private static CredenzialeMittente _getCredenzialeMittente(IDSoggetto dominio, String modulo, String idTransazione, 
    		Date scadenzaEntry,
    		AbstractSearchCredenziale searchCredential, AbstractCredenziale credential,
    		IOpenSPCoopState openspcoopState, String identitaChiamante,
    		StringBuilder sbConflict) throws Exception {
    	DBTransazioniManager dbManager = null;
    	Resource r = null;
    	boolean useConnectionRuntime = false;
		try{
			Connection con = null;
			
			OpenSPCoop2Properties openspcoopProperties = OpenSPCoop2Properties.getInstance();
			if(openspcoopProperties.isTransazioniUsePddRuntimeDatasource()) {
				if(openspcoopState!=null) {
					if(openspcoopState instanceof OpenSPCoopState) {
						OpenSPCoopState s = (OpenSPCoopState) openspcoopState;
						if(s.getConnectionDB()!=null && !s.getConnectionDB().isClosed()) {
							con = s.getConnectionDB();
							useConnectionRuntime = true;
						}
					}
				}
			}
			
			//boolean isNull = (openspcoopState==null);
			//System.out.println("GET CREDENZIALI MITTENTE useConnectionRuntime["+useConnectionRuntime+"] (isNull:"+isNull+") (identitaChiamante: "+identitaChiamante+") ("+idTransazione+") ...");
			
			if(useConnectionRuntime==false){
				dbManager = DBTransazioniManager.getInstance();
				r = dbManager.getResource(dominio, modulo+".credenzialeMittente", idTransazione);
				if(r==null){
					throw new Exception("Risorsa al database non disponibile");
				}
				con = (Connection) r.getResource();
			}
			
			if(con == null)
				throw new Exception("Connessione non disponibile");	
			//System.out.println("GET CREDENZIALI MITTENTE useConnectionRuntime["+useConnectionRuntime+"] (isNull:"+isNull+") (identitaChiamante: "+identitaChiamante+") ("+idTransazione+") OK");
			boolean debug = OpenSPCoop2Properties.getInstance().isTransazioniDebug();
			Logger logSql =  OpenSPCoop2Logger.getLoggerOpenSPCoopTransazioniSql(debug);
			DAOFactory daoFactory = DAOFactory.getInstance(logSql);
			DAOFactoryProperties daoFactoryProperties = DAOFactoryProperties.getInstance(logSql);
			ServiceManagerProperties daoFactoryServiceManagerPropertiesTransazioni = daoFactoryProperties.getServiceManagerProperties(org.openspcoop2.core.transazioni.utils.ProjectInfo.getInstance());
			daoFactoryServiceManagerPropertiesTransazioni.setShowSql(debug);
			daoFactoryServiceManagerPropertiesTransazioni.setDatabaseType(DBTransazioniManager.getInstance().getTipoDatabase());
			
			org.openspcoop2.core.transazioni.dao.IServiceManager transazioniSM = 
					(org.openspcoop2.core.transazioni.dao.IServiceManager) 
						daoFactory.getServiceManager(org.openspcoop2.core.transazioni.utils.ProjectInfo.getInstance(), con, daoFactoryServiceManagerPropertiesTransazioni, logSql);	

			ICredenzialeMittenteService credenzialiMittenteService = transazioniSM.getCredenzialeMittenteService();
			boolean ricercaEsatta = true;
	    	boolean caseSensitive = true;
	    	IPaginatedExpression pagExpression = searchCredential.createExpression(credenzialiMittenteService, credential.getCredenziale(), ricercaEsatta, caseSensitive);
			List<CredenzialeMittente> list = credenzialiMittenteService.findAll(pagExpression);
			if(list!=null && !list.isEmpty() && credential instanceof CredenzialeTrasporto) {
				CredenzialeTrasporto cTrasporto = (CredenzialeTrasporto) credential;
				if(cTrasporto.isSsl()) {
					list = CredenzialeSearchTrasporto.filterList(list, credential.getCredenziale(), logSql);
				}
			}

			if(list==null || list.size()<=0) {
				// not exists
				return createCredenzialeMittente(credenzialiMittenteService, scadenzaEntry, searchCredential, credential, logSql);
			}
			else if(list.size()>1) {
				throw new Exception("Trovata più di un'occorrenza di credenziale di tipo '"+searchCredential.getTipo()+"'; credenziale: ["+credential.getCredenziale()+"]");
			}
			else {
				CredenzialeMittente credenziale = list.get(0);
				if(credenziale.getOraRegistrazione().after(scadenzaEntry)) {
					return credenziale; // informazione in cache valida
				}
				else {
					return updateDataCredenzialeMittente(credenzialiMittenteService, scadenzaEntry, credenziale, sbConflict);
				}
			}
			
		}finally{
			if(useConnectionRuntime==false) {
				try{
					if(r!=null)
						dbManager.releaseResource(dominio, modulo, r);
				}catch(Exception eClose){
					// close
				}
			}
		}
    }
    
    private static CredenzialeMittente createCredenzialeMittente(ICredenzialeMittenteService credenzialeMittentiService,
    		Date scadenzaEntry,
    		AbstractSearchCredenziale searchCredential, AbstractCredenziale credential,
    		Logger log) throws Exception {
    	GestoreAutenticazione.lockCredenzialiMittenteCreazione.acquire("createCredenzialeMittente");
    	try {
    		return  GestoreAutenticazione._createCredenzialeMittente(credenzialeMittentiService,
    	    		scadenzaEntry,
    	    		searchCredential, credential,
    	    		log);
    	}finally {
    		GestoreAutenticazione.lockCredenzialiMittenteCreazione.release("createCredenzialeMittente");
    	}
    }
//    private static synchronized CredenzialeMittente createCredenzialeMittente(ICredenzialeMittenteService credenzialeMittentiService,
//    		Date scadenzaEntry,
//    		AbstractSearchCredenziale searchCredential, AbstractCredenziale credential,
//    		Logger log) throws Exception {
    private static CredenzialeMittente _createCredenzialeMittente(ICredenzialeMittenteService credenzialeMittentiService,
    		Date scadenzaEntry,
    		AbstractSearchCredenziale searchCredential, AbstractCredenziale credential,
    		Logger log) throws Exception {
    	boolean ricercaEsatta = true;
    	boolean caseSensitive = true;
    	IPaginatedExpression pagEpression = searchCredential.createExpression(credenzialeMittentiService, credential.getCredenziale(), ricercaEsatta, caseSensitive); 
    	List<CredenzialeMittente> list = credenzialeMittentiService.findAll(pagEpression);
    	if(list!=null && !list.isEmpty() && credential instanceof CredenzialeTrasporto) {
			CredenzialeTrasporto cTrasporto = (CredenzialeTrasporto) credential;
			if(cTrasporto.isSsl()) {
				list = CredenzialeSearchTrasporto.filterList(list, credential.getCredenziale(), log);
			}
		}
		if(list==null || list.size()<=0) {

			// not exists
			CredenzialeMittente credenzialeMittente = new CredenzialeMittente();
			credenzialeMittente.setTipo(credential.getTipo());
			credenzialeMittente.setRefCredenziale(credential.getReferenceId());
			credenzialeMittente.setOraRegistrazione(DateManager.getDate());
			credenzialeMittente.setCredenziale(credential.getCredenziale());
			try{
				credenzialeMittentiService.create(credenzialeMittente);
			}catch(Throwable t){
				// L'errore potrebbe essere dovuto al fatto che un thread su un altro nodo (dove non può agire la sincronizzazione del metodo) ha già creata la stessa entry
				// Si avrebbe un errore simile al seguente: "org.openspcoop2.generic_project.exception.ServiceException: Create not completed: insertAndReturnGeneratedKey failed: ORA-00001: unique constraint (GOVWAY_ENTERPRISE.UNIQUE_CREDENZIALE_MITTENTE_1) violated"
				// Provo a vedere se adesso le credenziali esistono
				
				list = credenzialeMittentiService.findAll(pagEpression);
				if(list!=null && !list.isEmpty() && credential instanceof CredenzialeTrasporto) {
					CredenzialeTrasporto cTrasporto = (CredenzialeTrasporto) credential;
					if(cTrasporto.isSsl()) {
						list = CredenzialeSearchTrasporto.filterList(list, credential.getCredenziale(), log);
					}
				}

				if(list==null || list.size()<=0) {
					// not exists, rilancio eccezione originale
					throw new Exception(t.getMessage(),t);
				}
				else if(list.size()>1) {
					throw new Exception("Trovata più di un'occorrenza di credenziale di tipo '"+searchCredential.getTipo()+"'; credenziale: ["+credential.getCredenziale()+"]");
				}
				else {
					CredenzialeMittente credenziale = list.get(0);
					return credenziale; // appena creata dall'altro nodo.
				}
			}
			return credenzialeMittente;
		}
		else if(list.size()>1) {
			throw new Exception("Trovata più di un'occorrenza di credenziale di tipo '"+searchCredential.getTipo()+"'; credenziale: ["+credential.getCredenziale()+"]");
		}
		else {
			CredenzialeMittente credenziale = list.get(0);
			if(credenziale.getOraRegistrazione().after(scadenzaEntry)) {
				return credenziale; // informazione in cache valida
			}
			//else {
				//return updateDataCredenzialeMittente(credenzialeMittentiService, scadenzaEntry, credenziale);
				// ritorno la credenziale trovata sul database, che comunque è sicuramente stata "appena" creata da un altro nodo run. Altrimenti non arriverei a questo else
				// il metodo di updateDataCredenzialeMittente imposta un ulteriore lock che non si desidera "pagare"
				return credenziale;
			//}
		}
    }
    
    private static CredenzialeMittente updateDataCredenzialeMittente(ICredenzialeMittenteService credenzialeMittentiService,
    		Date scadenzaEntry, CredenzialeMittente credenziale,
    		StringBuilder sbConflict) throws Exception {
    	
		// Il semaforo utilizzato prima di effettuare l'update non basta, la modifica potrebbe essere effettuata da più nodi.
		
		long gestioneSerializableDB_AttesaAttiva = OpenSPCoop2Properties.getInstance().getGestioneSerializableDBAttesaAttiva();
		int gestioneSerializableDB_CheckInterval = OpenSPCoop2Properties.getInstance().getGestioneSerializableDBCheckInterval();
		
		long scadenzaWhile = DateManager.getTimeMillis() + gestioneSerializableDB_AttesaAttiva;
		
		Throwable lastT = null;
		while(DateManager.getTimeMillis() < scadenzaWhile){
		
			int conflict = 0;
			try{
				
				GestoreAutenticazione.lockCredenzialiMittenteAggiornamento.acquire("updateDataCredenzialeMittente");
				try {
					CredenzialeMittente c = GestoreAutenticazione._updateDataCredenzialeMittente(credenzialeMittentiService,
							scadenzaEntry, credenziale);
					if(sbConflict!=null) {
						sbConflict.append(conflict);
					}
					return c;
	    		
				}finally {
		    		GestoreAutenticazione.lockCredenzialiMittenteAggiornamento.release("updateDataCredenzialeMittente");
		    	}
		    		
			} catch(Throwable e) {
				lastT = e;
				//System.out.println("Serializable error:"+e.getMessage());
				//e.printStackTrace(System.out);
				conflict++;
			}
    		
			// Per aiutare ad evitare conflitti
			try{
				Utilities.sleep((getRandom()).nextInt(gestioneSerializableDB_CheckInterval)); // random da 0ms a checkIntervalms
			}catch(Exception eRandom){
				// ignore
			}
		}
    	
		if(lastT!=null) {
			throw new Exception("Aggiornamento credenziali non riuscito: "+lastT.getMessage(),lastT);
		}
		else {
			throw new Exception("Aggiornamento credenziali non riuscito");
		}
		
    }
//    private static synchronized CredenzialeMittente updateDataCredenzialeMittente(ICredenzialeMittenteService credenzialeMittentiService,
//    		Date scadenzaEntry, CredenzialeMittente credenziale) throws Exception {
    private static CredenzialeMittente _updateDataCredenzialeMittente(ICredenzialeMittenteService credenzialeMittentiService,
    		Date scadenzaEntry, CredenzialeMittente credenziale) throws Exception {
    	
    	IdCredenzialeMittente id = credenzialeMittentiService.convertToId(credenziale);
    	CredenzialeMittente credenzialeCheck = credenzialeMittentiService.get(id);
    	if(credenziale.getOraRegistrazione().after(scadenzaEntry)) {
			return credenzialeCheck; // informazione in cache già aggiornata da una chiamata concorrente
		}
		else {
			//System.out.println("Update Credenziale '"+id.getTipo()+"'");
			credenzialeCheck.setOraRegistrazione(DateManager.getDate()); // update
			credenzialeMittentiService.update(id, credenzialeCheck);
			return credenzialeCheck;
		}
    	
    }
    
}