InResponseHandler.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.handlers.transazioni;
import java.io.ByteArrayOutputStream;
import java.util.Date;
import org.apache.commons.lang.StringUtils;
import org.openspcoop2.core.constants.Costanti;
import org.openspcoop2.core.constants.TipoPdD;
import org.openspcoop2.message.OpenSPCoop2RestMessage;
import org.openspcoop2.message.OpenSPCoop2SoapMessage;
import org.openspcoop2.message.constants.MessageRole;
import org.openspcoop2.message.constants.ServiceBinding;
import org.openspcoop2.message.xml.MessageXMLUtils;
import org.openspcoop2.pdd.config.OpenSPCoop2Properties;
import org.openspcoop2.pdd.core.CostantiPdD;
import org.openspcoop2.pdd.core.connettori.ConnettoreBase;
import org.openspcoop2.pdd.core.handlers.HandlerException;
import org.openspcoop2.pdd.core.handlers.InResponseContext;
import org.openspcoop2.pdd.core.transazioni.InResponseStatefulObject;
import org.openspcoop2.pdd.core.transazioni.RepositoryGestioneStateful;
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.pdd.core.transazioni.TransactionStatefulNotSupportedException;
import org.openspcoop2.pdd.logger.DumpUtility;
import org.openspcoop2.pdd.logger.OpenSPCoop2Logger;
import org.openspcoop2.utils.json.JSONUtils;
import org.openspcoop2.utils.transport.http.HttpRequestMethod;
import org.slf4j.Logger;
import com.fasterxml.jackson.databind.JsonNode;
/**
* InResponseHandler
*
* @author Poli Andrea (poli@link.it)
* @author $Author$
* @version $Rev$, $Date$
*/
public class InResponseHandler extends FirstPositionHandler implements org.openspcoop2.pdd.core.handlers.InResponseHandler{
@Override
public void invoke(InResponseContext context) throws HandlerException {
OpenSPCoop2Properties op2Properties = OpenSPCoop2Properties.getInstance();
if(op2Properties.isTransazioniEnabled()==false) {
return;
}
String idTransazione = null;
if(context!=null && context.getPddContext()!=null) {
idTransazione = (String) context.getPddContext().getObject(Costanti.ID_TRANSAZIONE);
}
//System.out.println("------------- InResponseHandler ("+idTransazione+")("+context.getTipoPorta().getTipo()+") -------------------");
// Gestione FAULT
String fault = null;
String formatoFault = null;
try{
if(context!=null && context.getMessaggio()!=null){
if(ServiceBinding.SOAP.equals(context.getMessaggio().getServiceBinding())) {
OpenSPCoop2SoapMessage soapMsg = context.getMessaggio().castAsSoap();
if(soapMsg.hasSOAPFault()){
ByteArrayOutputStream bout = new ByteArrayOutputStream();
bout.write(context.getMessaggio().getAsByte(soapMsg.getSOAPPart().getEnvelope(), false));
bout.flush();
bout.close();
Logger log = OpenSPCoop2Logger.getLoggerOpenSPCoopTransazioni(op2Properties.isTransazioniDebug());
if(op2Properties.isTransazioniFaultPrettyPrint()){
// Faccio una pretty-print: potevo fare anche direttamente passando il fault a metodo prettyPrint,
// Pero' non veniva stampato correttamente il SOAPFault. Mi appoggio allora a SoapUtils.
//byte [] content = org.openspcoop2.message.soap.TunnelSoapUtils.sbustamentoMessaggio(context.getMessaggio());
byte [] content = bout.toByteArray();
fault = DumpUtility.toString(MessageXMLUtils.getInstance(soapMsg.getFactory()).newDocument(content), log, context.getMessaggio());
//System.out.println("IMPOSTATO FAULT IN TRANSACTION ["+fault+"]");
}
else{
fault = bout.toString();
}
formatoFault = soapMsg.getMessageType().name();
}
}
else {
OpenSPCoop2RestMessage<?> restMsg = context.getMessaggio().castAsRest();
if(restMsg.isProblemDetailsForHttpApis_RFC7807() || MessageRole.FAULT.equals(restMsg.getMessageRole())) {
switch (restMsg.getMessageType()) {
case XML:
ByteArrayOutputStream bout = new ByteArrayOutputStream();
restMsg.writeTo(bout, false);
bout.flush();
bout.close();
Logger log = OpenSPCoop2Logger.getLoggerOpenSPCoopTransazioni(op2Properties.isTransazioniDebug());
if(op2Properties.isTransazioniFaultPrettyPrint()){
// Faccio una pretty-print: potevo fare anche direttamente passando il fault a metodo prettyPrint,
// Pero' non veniva stampato correttamente il SOAPFault. Mi appoggio allora a SoapUtils.
//byte [] content = org.openspcoop2.message.soap.TunnelSoapUtils.sbustamentoMessaggio(context.getMessaggio());
byte [] content = bout.toByteArray();
fault = DumpUtility.toString(MessageXMLUtils.getInstance(restMsg.getFactory()).newDocument(content), log, context.getMessaggio());
//System.out.println("IMPOSTATO FAULT IN TRANSACTION ["+fault+"]");
}
else{
fault = bout.toString();
}
formatoFault = restMsg.getMessageType().name();
break;
case JSON:
bout = new ByteArrayOutputStream();
restMsg.writeTo(bout, false);
bout.flush();
bout.close();
if(op2Properties.isTransazioniFaultPrettyPrint()){
JSONUtils jsonUtils = JSONUtils.getInstance(true);
byte [] content = bout.toByteArray();
JsonNode jsonNode = jsonUtils.getAsNode(content);
fault = jsonUtils.toString(jsonNode);
}
else{
fault = bout.toString();
}
formatoFault = restMsg.getMessageType().name();
break;
default:
break;
}
}
}
}
}catch(Exception e){
throw new HandlerException("Errore durante il dump del soap fault",e);
}
if(context!=null && context.getTransazioneApplicativoServer()!=null) {
try{
// date
context.getTransazioneApplicativoServer().setDataAccettazioneRisposta(context.getDataAccettazioneRisposta());
Date dataIngressoRisposta = context.getDataElaborazioneMessaggio();
if(context.getDataTerminataInvocazioneConnettore()!=null) {
dataIngressoRisposta = context.getDataTerminataInvocazioneConnettore(); // nella latenza porta deve rientrare anche la negoziazione della connessione
}
context.getTransazioneApplicativoServer().setDataIngressoRisposta(dataIngressoRisposta);
if(context.getDataPrimaInvocazioneConnettore()!=null) {
// aggiorno informazione sulla data, poiche' piu' precisa (nella latenza porta deve rientrare anche il dump ed il rilascio della connessione)
context.getTransazioneApplicativoServer().setDataUscitaRichiesta(context.getDataPrimaInvocazioneConnettore());
}
if(context.getDataRichiestaInoltrata()!=null) {
context.getTransazioneApplicativoServer().setDataUscitaRichiestaStream(context.getDataRichiestaInoltrata());
}
// return code
context.getTransazioneApplicativoServer().setCodiceRisposta(context.getReturnCode()+"");
// eventuali errori
context.getTransazioneApplicativoServer().setUltimoErrore(context.getErroreConsegna());
// aggiorno location
if(context.getConnettore()!=null){
context.getTransazioneApplicativoServer().setLocationConnettore(context.getConnettore().getLocation());
}
// fault
if(fault!=null){
context.getTransazioneApplicativoServer().setFault(fault);
context.getTransazioneApplicativoServer().setFormatoFault(formatoFault);
}
}catch(Exception e){
throw new HandlerException("Errore durante il processamento delle informazioni relative alla consegna per l'applicativo '"+context.getTransazioneApplicativoServer().getServizioApplicativoErogatore()+"': "+e.getMessage(),e);
}
}
else {
boolean gestioneStateful = false;
Transaction tr = null;
try{
tr = TransactionContext.getTransaction(idTransazione);
}catch(TransactionNotExistsException e){
gestioneStateful = true;
}
try{
InResponseStatefulObject sObject = null;
if(tr==null && gestioneStateful){
sObject = new InResponseStatefulObject();
//System.out.println("@@@@@REPOSITORY@@@@@ InResponseHandler ID TRANSAZIONE ["+idTransazione+"] GESTIONE COMPLETA");
Date dataRichiestaInoltrata = context.getDataRichiestaInoltrata();
// La porta di dominio mi passa sempre questa informazione.
// Nel PddMonitor, invece, la data deve essere visualizzata solo se la dimensione e' diverso da 0 e cioe' se c'e' un messaggio di risposta.
//if(dimensione!=null && dimensione>0){
// L'INFORMAZIONE DEVE INVECE ESSERE SEMPRE SALVATA PER LA SIMULAZIONE DEI MESSAGGI DIAGNOSTICI
// INEFFICIENTE: RepositoryGestioneStateful.addDataAccettazioneRisposta(idTransazione, dataAccettazioneRisposta);
sObject.setDataRichiestaInoltrata(dataRichiestaInoltrata);
Date dataAccettazioneRisposta = context.getDataAccettazioneRisposta();
// La porta di dominio mi passa sempre questa informazione.
// Nel PddMonitor, invece, la data deve essere visualizzata solo se la dimensione e' diverso da 0 e cioe' se c'e' un messaggio di risposta.
//if(dimensione!=null && dimensione>0){
// L'INFORMAZIONE DEVE INVECE ESSERE SEMPRE SALVATA PER LA SIMULAZIONE DEI MESSAGGI DIAGNOSTICI
// INEFFICIENTE: RepositoryGestioneStateful.addDataAccettazioneRisposta(idTransazione, dataAccettazioneRisposta);
sObject.setDataAccettazioneRisposta(dataAccettazioneRisposta);
Date dataIngressoRisposta = context.getDataElaborazioneMessaggio();
if(context.getDataTerminataInvocazioneConnettore()!=null) {
dataIngressoRisposta = context.getDataTerminataInvocazioneConnettore(); // nella latenza porta deve rientrare anche la negoziazione della connessione
}
// La porta di dominio mi passa sempre questa informazione.
// Nel PddMonitor, invece, la data deve essere visualizzata solo se la dimensione e' diverso da 0 e cioe' se c'e' un messaggio di risposta.
//if(dimensione!=null && dimensione>0){
// L'INFORMAZIONE DEVE INVECE ESSERE SEMPRE SALVATA PER LA SIMULAZIONE DEI MESSAGGI DIAGNOSTICI
// INEFFICIENTE: RepositoryGestioneStateful.addDataIngressoRisposta(idTransazione, dataElaborazioneMessaggio);
sObject.setDataIngressoRisposta(dataIngressoRisposta);
if(context.getDataPrimaInvocazioneConnettore()!=null) {
// aggiorno informazione sulla data, poiche' piu' precisa (nella latenza porta deve rientrare anche il dump ed il rilascio della connessione)
sObject.setDataUscitaRichiesta(context.getDataPrimaInvocazioneConnettore());
}
// INEFFICIENTE: RepositoryGestioneStateful.addCodiceTrasportoRichiesta(idTransazione, context.getReturnCode()+"");
sObject.setReturnCode(context.getReturnCode()+"");
if(context.getConnettore()!=null){
// INEFFICIENTE: RepositoryGestioneStateful.addLocation(idTransazione, context.getConnettore().getLocation());
sObject.setLocation(context.getConnettore().getLocation());
}
if(fault!=null){
if(TipoPdD.APPLICATIVA.equals(context.getTipoPorta())){
sObject.setFaultIntegrazione(fault);
sObject.setFormatoFaultIntegrazione(formatoFault);
}
else{
sObject.setFaultCooperazione(fault);
sObject.setFormatoFaultCooperazione(formatoFault);
}
}
}else{
if(tr==null) {
throw new HandlerException("Transaction is null");
}
if(context==null) {
throw new HandlerException("Context is null");
}
Date dataRichiestaInoltrata = context.getDataRichiestaInoltrata();
// La porta di dominio mi passa sempre questa informazione.
// Nel PddMonitor, invece, la data deve essere visualizzata solo se la dimensione e' diverso da 0 e cioe' se c'e' un messaggio di risposta.
//if(dimensione!=null && dimensione>0){
// L'INFORMAZIONE DEVE INVECE ESSERE SEMPRE SALVATA PER LA SIMULAZIONE DEI MESSAGGI DIAGNOSTICI
try{
tr.setDataRichiestaInoltrata(dataRichiestaInoltrata);
//System.out.println("SET DATA ("+dataAccettazioneRisposta.toString()+")");
}catch(TransactionDeletedException e){
//System.out.println("@@@@@REPOSITORY@@@@@ InResponseHandler SET DATA ("+dataAccettazioneRisposta.toString()+")");
// INEFFICIENTE: RepositoryGestioneStateful.addDataAccettazioneRisposta(idTransazione, dataAccettazioneRisposta);
if(sObject==null)
sObject = new InResponseStatefulObject();
sObject.setDataRichiestaInoltrata(dataRichiestaInoltrata);
}
Date dataAccettazioneRisposta = context.getDataAccettazioneRisposta();
// La porta di dominio mi passa sempre questa informazione.
// Nel PddMonitor, invece, la data deve essere visualizzata solo se la dimensione e' diverso da 0 e cioe' se c'e' un messaggio di risposta.
//if(dimensione!=null && dimensione>0){
// L'INFORMAZIONE DEVE INVECE ESSERE SEMPRE SALVATA PER LA SIMULAZIONE DEI MESSAGGI DIAGNOSTICI
try{
tr.setDataAccettazioneRisposta(dataAccettazioneRisposta);
//System.out.println("SET DATA ("+dataAccettazioneRisposta.toString()+")");
}catch(TransactionDeletedException e){
//System.out.println("@@@@@REPOSITORY@@@@@ InResponseHandler SET DATA ("+dataAccettazioneRisposta.toString()+")");
// INEFFICIENTE: RepositoryGestioneStateful.addDataAccettazioneRisposta(idTransazione, dataAccettazioneRisposta);
if(sObject==null)
sObject = new InResponseStatefulObject();
sObject.setDataAccettazioneRisposta(dataAccettazioneRisposta);
}
Date dataIngressoRisposta = context.getDataElaborazioneMessaggio();
if(context.getDataTerminataInvocazioneConnettore()!=null) {
dataIngressoRisposta = context.getDataTerminataInvocazioneConnettore(); // nella latenza porta deve rientrare anche la negoziazione della connessione
}
// La porta di dominio mi passa sempre questa informazione.
// Nel PddMonitor, invece, la data deve essere visualizzata solo se la dimensione e' diverso da 0 e cioe' se c'e' un messaggio di risposta.
//if(dimensione!=null && dimensione>0){
// L'INFORMAZIONE DEVE INVECE ESSERE SEMPRE SALVATA PER LA SIMULAZIONE DEI MESSAGGI DIAGNOSTICI
try{
tr.setDataIngressoRisposta(dataIngressoRisposta);
//System.out.println("SET DATA ("+dataElaborazioneMessaggio.toString()+")");
}catch(TransactionDeletedException e){
//System.out.println("@@@@@REPOSITORY@@@@@ InResponseHandler SET DATA ("+dataElaborazioneMessaggio.toString()+")");
// INEFFICIENTE: RepositoryGestioneStateful.addDataIngressoRisposta(idTransazione, dataElaborazioneMessaggio);
if(sObject==null)
sObject = new InResponseStatefulObject();
sObject.setDataIngressoRisposta(dataIngressoRisposta);
}
if(context.getDataPrimaInvocazioneConnettore()!=null) {
// aggiorno informazione sulla data, poiche' piu' precisa (nella latenza porta deve rientrare anche il dump ed il rilascio della connessione)
try{
tr.setDataUscitaRichiesta(context.getDataPrimaInvocazioneConnettore());
//System.out.println("SET DATA ("+dataElaborazioneMessaggio.toString()+")");
}catch(TransactionDeletedException e){
//System.out.println("@@@@@REPOSITORY@@@@@ InResponseHandler SET DATA ("+dataElaborazioneMessaggio.toString()+")");
// INEFFICIENTE: RepositoryGestioneStateful.addDataIngressoRisposta(idTransazione, dataElaborazioneMessaggio);
if(sObject==null)
sObject = new InResponseStatefulObject();
sObject.setDataUscitaRichiesta(context.getDataPrimaInvocazioneConnettore());
}
}
try{
//System.out.println("SET CODICE TRASPORTO RICHIESTA ["+context.getReturnCode()+"]");
if(context!=null) {
tr.setCodiceTrasportoRichiesta(context.getReturnCode()+"");
}
}catch(TransactionDeletedException e){
//System.out.println("@@@@@REPOSITORY@@@@@ InResponseHandler SET CODICE TRASPORTO RICHIESTA ["+context.getReturnCode()+"]");
// INEFFICIENTE: RepositoryGestioneStateful.addCodiceTrasportoRichiesta(idTransazione, context.getReturnCode()+"");
if(sObject==null)
sObject = new InResponseStatefulObject();
sObject.setReturnCode(context.getReturnCode()+"");
}
try{
// update location impostata nel OutRequest (location modificata dal connettore)
//System.out.println("SET LOCATION ["+context.getConnettore().getLocation()+"]");
String connettoreRequestUrl = null;
String connettoreRequestMethod = null;
if(context.getPddContext()!=null) {
if(context.getPddContext().containsKey(CostantiPdD.CONNETTORE_REQUEST_URL)) {
connettoreRequestUrl = (String) context.getPddContext().getObject(CostantiPdD.CONNETTORE_REQUEST_URL);
}
if(context.getPddContext().containsKey(CostantiPdD.CONNETTORE_REQUEST_METHOD)) {
Object o = context.getPddContext().getObject(CostantiPdD.CONNETTORE_REQUEST_METHOD);
if(o instanceof String) {
connettoreRequestMethod = (String) o;
}
else if(o instanceof HttpRequestMethod) {
HttpRequestMethod oConnettoreRequestMethod = (HttpRequestMethod) o;
connettoreRequestMethod = oConnettoreRequestMethod.name();
}
}
}
if(!StringUtils.isEmpty(connettoreRequestUrl) && !StringUtils.isEmpty(connettoreRequestMethod)) {
String prefix = "";
if(ConnettoreBase.LOCATION_CACHED.equals(context.getConnettore().getLocation())){
prefix = context.getConnettore().getLocation()+ConnettoreBase.LOCATION_CACHED_SEPARATOR_REQUEST_URL;
}
tr.setLocation(prefix+CostantiPdD.getConnettoreRequest(connettoreRequestUrl, connettoreRequestMethod));
}
else {
if(context!=null && context.getConnettore()!=null) {
tr.setLocation(context.getConnettore().getLocation());
}
}
}catch(TransactionDeletedException e){
//System.out.println("@@@@@REPOSITORY@@@@@ OutRequestHandler SET LOCATION ["+context.getConnettore().getLocation()+"]");
// INEFFICIENTE: RepositoryGestioneStateful.addLocation(idTransazione, context.getConnettore().getLocation());
if(sObject==null)
sObject = new InResponseStatefulObject();
sObject.setLocation(context.getConnettore().getLocation());
}
try{
if(fault!=null){
if(TipoPdD.APPLICATIVA.equals(context.getTipoPorta())){
tr.setFaultIntegrazione(fault);
tr.setFormatoFaultIntegrazione(formatoFault);
}
else{
tr.setFaultCooperazione(fault);
tr.setFormatoFaultCooperazione(formatoFault);
}
}
}catch(TransactionDeletedException e){
if(fault!=null){
if(sObject==null)
sObject = new InResponseStatefulObject();
if(TipoPdD.APPLICATIVA.equals(context.getTipoPorta())){
sObject.setFaultIntegrazione(fault);
sObject.setFormatoFaultIntegrazione(formatoFault);
}
else{
sObject.setFaultCooperazione(fault);
sObject.setFormatoFaultCooperazione(formatoFault);
}
}
}
}
if(sObject!=null){
// Gestione stateful
RepositoryGestioneStateful.addInResponseStatefulObject(context.getProtocolFactory().getProtocol(),idTransazione, sObject);
}
}catch(TransactionStatefulNotSupportedException e){
throw new HandlerException("Errore durante il processamento dell'handler: "+e.getMessage(),e);
}
}
}
}