TimerEventiThread.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.timers;


import java.io.File;
import java.io.FileInputStream;
import java.sql.Connection;
import java.util.Date;
import java.util.List;

import org.openspcoop2.core.controllo_traffico.driver.PolicyGroupByActiveThreadsType;
import org.openspcoop2.pdd.config.DBTransazioniManager;
import org.openspcoop2.pdd.config.OpenSPCoop2Properties;
import org.openspcoop2.pdd.config.Resource;
import org.openspcoop2.pdd.core.controllo_traffico.NotificatoreEventi;
import org.openspcoop2.pdd.core.controllo_traffico.policy.driver.GestorePolicyAttive;
import org.openspcoop2.pdd.core.handlers.HandlerException;
import org.openspcoop2.utils.UtilsException;
import org.openspcoop2.utils.date.DateManager;
import org.openspcoop2.utils.threads.BaseThread;
import org.slf4j.Logger;


/**     
 * TimerEventiThread
 *
 * @author Poli Andrea (poli@link.it)
 * @author $Author$
 * @version $Rev$, $Date$
 */
public class TimerEventiThread extends BaseThread{

	private static TimerState STATE = TimerState.OFF; // abilitato in OpenSPCoop2Startup al momento dell'avvio
	
	public static TimerState getSTATE() {
		return STATE;
	}
	public static void setSTATE(TimerState sTATE) {
		STATE = sTATE;
	}

	public static final String ID_MODULO = "TimerEventi";
		
	/** Logger utilizzato per debug. */
	private Logger log = null;
	
	/** Indicazione se deve essere effettuato il log delle query */
	private boolean debug = false;	
	
	
	private OpenSPCoop2Properties properties;
	
	/** NotificatoreEventi */
	private NotificatoreEventi notificatoreEventi = null;
	
	/** LastInterval */
	private Date lastInterval;
	
	/** ConnectionTimeout */
	private int checkConnectionTimeoutEveryXTimes = 1;
	private int offsetConnectionTimeoutEveryXTimes = 0;
	private Date lastIntervalConnectionTimeout;
	
	/** RequestReadTimeout */
	private int checkRequestReadTimeoutEveryXTimes = 1;
	private int offsetRequestReadTimeoutEveryXTimes = 0;
	private Date lastIntervalRequestReadTimeout;
	
	private int checkReadTimeoutEveryXTimes = 1;
	private int offsetReadTimeoutEveryXTimes = 0;
	private Date lastIntervalReadTimeout;
	
	/** Immagine */
	@SuppressWarnings("unused")
	private boolean forceCheckPrimoAvvio = false;
	
	private static boolean inizializzazioneAttiva = false;
	public static boolean isInizializzazioneAttiva() {
		return inizializzazioneAttiva;
	}
	public static void setInizializzazioneAttiva(boolean inizializzazioneAttiva) {
		TimerEventiThread.inizializzazioneAttiva = inizializzazioneAttiva;
	}
	
	/** Costruttore */
	@SuppressWarnings("deprecation")
	public TimerEventiThread(Logger log) throws Exception{
	
		this.log = log;
	
		this.properties = OpenSPCoop2Properties.getInstance();
		this.setTimeout(this.properties.getEventiTimerIntervalSeconds());
		
		this.checkConnectionTimeoutEveryXTimes = this.properties.getEventiTimerIntervalConnectionTimeoutEveryXTimes();
		this.checkRequestReadTimeoutEveryXTimes = this.properties.getEventiTimerIntervalRequestReadTimeoutEveryXTimes();
		this.checkReadTimeoutEveryXTimes = this.properties.getEventiTimerIntervalReadTimeoutEveryXTimes();
		
		// Eventi per Controllo Traffico
		if(this.properties.isControlloTrafficoEnabled()){
			this.notificatoreEventi = NotificatoreEventi.getInstance();
			
			// Il meccanismo di ripristino dell'immagine degli eventi non sembra funzionare
			// Lascio comunque il codice se in futuro si desidera approfindire la questione
			if(inizializzazioneAttiva) {
				List<PolicyGroupByActiveThreadsType> tipiGestorePolicyRateLimiting = null;
				try{
					tipiGestorePolicyRateLimiting = GestorePolicyAttive.getTipiGestoriAttivi();
				}catch(Throwable e){
					this.log.error("Errore durante l'inizializzazione dell'immagine degli eventi per il Controllo del Traffico: "+e.getMessage(),e);
				}
				if(tipiGestorePolicyRateLimiting!=null && !tipiGestorePolicyRateLimiting.isEmpty()) {
					for (PolicyGroupByActiveThreadsType type : tipiGestorePolicyRateLimiting) {
						File fDati = null;
						try{
							File fRepository = this.properties.getControlloTrafficoGestorePolicyFileSystemRecoveryRepository();
							if(fRepository!=null){
								if(fRepository.exists()==false){
									throw new Exception("Directory ["+fRepository.getAbsolutePath()+"] not exists");
								}
								if(fRepository.isDirectory()==false){
									throw new Exception("File ["+fRepository.getAbsolutePath()+"] is not directory");
								}
								if(fRepository.canRead()==false){
									throw new Exception("File ["+fRepository.getAbsolutePath()+"] cannot read");
								}
								if(fRepository.canWrite()==false){
									throw new Exception("File ["+fRepository.getAbsolutePath()+"] cannot write");
								}
								fDati = new File(fRepository, GestorePolicyAttive.getControlloTrafficoEventiImage(type));
								if(fDati.exists() && fDati.canRead() && fDati.length()>0){
									FileInputStream fin = new FileInputStream(fDati);
									this.notificatoreEventi.initialize(fin);
									if(!fDati.delete()) {
										// ignore
									}
									this.forceCheckPrimoAvvio = true;
								}
							}
						}catch(Exception e){
							String img = null;
							if(fDati!=null){
								img = fDati.getAbsolutePath();
							}
							throw new HandlerException("Inizializzazione dell'immagine degli eventi ["+img+"] per il Controllo del Traffico non riuscita: "+e.getMessage(),e);
						}
					}
				}
			}
		}
		
		this.lastInterval = DateManager.getDate();
		this.lastIntervalConnectionTimeout = DateManager.getDate();
		this.lastIntervalRequestReadTimeout = DateManager.getDate();
		this.lastIntervalReadTimeout = DateManager.getDate();
		
		this.debug = this.properties.isEventiDebug();

	}
	
	
	@Override
	public void process(){
	
		if(TimerState.ENABLED.equals(STATE)) {
		
			DBTransazioniManager dbManager = null;
	    	Resource r = null;
	    	try{
	    		dbManager = DBTransazioniManager.getInstance();
				r = dbManager.getResource(this.properties.getIdentitaPortaDefaultWithoutProtocol(), ID_MODULO, null);
				if(r==null){
					throw new UtilsException("Risorsa al database non disponibile");
				}
				Connection con = (Connection) r.getResource();
				if(con == null)
					throw new UtilsException("Connessione non disponibile");	
				
				if(this.properties.isControlloTrafficoEnabled()){
					
					try{
						this.lastInterval = this.notificatoreEventi.process(this.log, this.getTimeout(), this.lastInterval, con, this.debug);
					}catch(Exception e){
						this.log.error("Errore durante la generazione degli eventi per il controllo del traffico: "+e.getMessage(),e);
					}
					
					// Comprensione offset per analisi connection timeout events
					try{
						boolean analyzeConnectionTimeout = isAnalyzeConnectionTimeout();
						if(analyzeConnectionTimeout) {
							this.lastIntervalConnectionTimeout = this.notificatoreEventi.processConnectionTimeout(this.log, (this.getTimeout()*this.checkConnectionTimeoutEveryXTimes), this.lastIntervalConnectionTimeout, con, this.debug);
						}
						else {
							this.notificatoreEventi.emitProcessConnectionTimeoutSkip(this.log, this.debug, this.offsetConnectionTimeoutEveryXTimes, this.checkConnectionTimeoutEveryXTimes);
						}
					}catch(Exception e){
						this.log.error("Errore durante la generazione degli eventi di connection timeout: "+e.getMessage(),e);
					}
					
					// Comprensione offset per analisi request read timeout events
					try {
						boolean analyzeRequestReadTimeout = isAnalyzeRequestReadTimeout();
						if(analyzeRequestReadTimeout) {
							this.lastIntervalRequestReadTimeout = this.notificatoreEventi.processRequestReadTimeout(this.log, (this.getTimeout()*this.checkRequestReadTimeoutEveryXTimes), this.lastIntervalRequestReadTimeout, con, this.debug);
						}
						else {
							this.notificatoreEventi.emitProcessRequestReadTimeoutSkip(this.log, this.debug, this.offsetRequestReadTimeoutEveryXTimes, this.checkRequestReadTimeoutEveryXTimes);
						}
					}catch(Exception e){
						this.log.error("Errore durante la generazione degli eventi di request read timeout: "+e.getMessage(),e);
					}
					
					// Comprensione offset per analisi read timeout events
					try {
						boolean analyzeReadTimeout = isAnalyzeReadTimeout();
						if(analyzeReadTimeout) {
							this.lastIntervalReadTimeout = this.notificatoreEventi.processReadTimeout(this.log, (this.getTimeout()*this.checkReadTimeoutEveryXTimes), this.lastIntervalReadTimeout, con, this.debug);
						}
						else {
							this.notificatoreEventi.emitProcessReadTimeoutSkip(this.log, this.debug, this.offsetReadTimeoutEveryXTimes, this.checkReadTimeoutEveryXTimes);
						}
					}catch(Exception e){
						this.log.error("Errore durante la generazione degli eventi di request read timeout: "+e.getMessage(),e);
					}
					
				}
								
				// Aggiungere in futuro altre gestione degli eventi
				
			}catch(Exception e){
				this.log.error("Errore durante la generazione degli eventi: "+e.getMessage(),e);
			}finally{
				try{
					if(r!=null)
						dbManager.releaseResource(this.properties.getIdentitaPortaDefaultWithoutProtocol(), ID_MODULO, r);
				}catch(Exception eClose){
					// ignore
				}
			}
	    	
		}
		else {
			this.log.info("Timer "+ID_MODULO+" disabilitato");
		}
				
	}
	
	@Override
	public void close(){
		this.log.info("Thread per la generazione degli eventi terminato");
	}
			
	private boolean isAnalyzeConnectionTimeout() {
		this.offsetConnectionTimeoutEveryXTimes++;
		boolean esito = false;
		if(this.offsetConnectionTimeoutEveryXTimes==this.checkConnectionTimeoutEveryXTimes) {
			/**System.out.println("CONNECTION TIMEOUT CHECK '"+this.offsetConnectionTimeoutEveryXTimes+"'=='"+this.checkConnectionTimeoutEveryXTimes+"' TRUE");*/
			esito = true;
			this.offsetConnectionTimeoutEveryXTimes = 0;
		}
		/**else {
			System.out.println("CONNECTION TIMEOUT CHECK '"+this.offsetConnectionTimeoutEveryXTimes+"'<>'"+this.checkConnectionTimeoutEveryXTimes+"' FALSE");
		}*/
		return esito;
	}
	private boolean isAnalyzeRequestReadTimeout() {
		this.offsetRequestReadTimeoutEveryXTimes++;
		boolean esito = false;
		if(this.offsetRequestReadTimeoutEveryXTimes==this.checkRequestReadTimeoutEveryXTimes) {
			/**System.out.println("REQUEST READ TIMEOUT CHECK '"+this.offsetRequestReadTimeoutEveryXTimes+"'=='"+this.checkRequestReadTimeoutEveryXTimes+"' TRUE");*/
			esito = true;
			this.offsetRequestReadTimeoutEveryXTimes = 0;
		}
		/**else {
			System.out.println("REQUEST READ TIMEOUT CHECK '"+this.offsetRequestReadTimeoutEveryXTimes+"'<>'"+this.checkRequestReadTimeoutEveryXTimes+"' FALSE");
		}*/
		return esito;
	}
	private boolean isAnalyzeReadTimeout() {
		this.offsetReadTimeoutEveryXTimes++;
		boolean esito = false;
		if(this.offsetReadTimeoutEveryXTimes==this.checkReadTimeoutEveryXTimes) {
			/**System.out.println("READ TIMEOUT CHECK '"+this.offsetReadTimeoutEveryXTimes+"'=='"+this.checkReadTimeoutEveryXTimes+"' TRUE");*/
			esito = true;
			this.offsetReadTimeoutEveryXTimes = 0;
		}
		else {
			/**System.out.println("READ TIMEOUT CHECK '"+this.offsetReadTimeoutEveryXTimes+"'<>'"+this.checkReadTimeoutEveryXTimes+"' FALSE");*/
		}
		return esito;
	}
}