io.tracee.contextlogger.contextprovider.jaxws.AbstractTraceeErrorLoggingHandler Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of contextprovider-jaxws Show documentation
Show all versions of contextprovider-jaxws Show documentation
Please refer to https://github.com/tracee/contextlogger.
package io.tracee.contextlogger.contextprovider.jaxws;
import java.io.ByteArrayOutputStream;
import java.nio.charset.Charset;
import java.util.Set;
import javax.xml.namespace.QName;
import javax.xml.soap.SOAPMessage;
import javax.xml.ws.handler.MessageContext;
import javax.xml.ws.handler.soap.SOAPHandler;
import javax.xml.ws.handler.soap.SOAPMessageContext;
import io.tracee.Tracee;
import io.tracee.TraceeBackend;
import io.tracee.TraceeLogger;
import io.tracee.contextlogger.MessagePrefixProvider;
import io.tracee.contextlogger.TraceeContextLogger;
import io.tracee.contextlogger.api.ImplicitContext;
import io.tracee.contextlogger.api.internal.MessageLogLevel;
import io.tracee.contextlogger.contextprovider.jaxws.contextprovider.JaxWsWrapper;
/**
* Abstract base class for detecting JAX-WS related uncaught exceptions and outputting of contextual information.
*/
public abstract class AbstractTraceeErrorLoggingHandler implements SOAPHandler {
protected final TraceeBackend traceeBackend;
private final TraceeLogger logger;
protected static final ThreadLocal THREAD_LOCAL_SOAP_MESSAGE_STR = new ThreadLocal();
/**
* The constructor.
*
* @param traceeBackend the tracee backend to use
*/
AbstractTraceeErrorLoggingHandler(TraceeBackend traceeBackend) {
this.traceeBackend = traceeBackend;
logger = traceeBackend.getLoggerFactory().getLogger(AbstractTraceeErrorLoggingHandler.class);
}
/**
* No-Arg Constructor
*/
public AbstractTraceeErrorLoggingHandler() {
this(Tracee.getBackend());
}
@Override
public final boolean handleMessage(final SOAPMessageContext context) {
if (this.isOutgoing(context)) {
this.handleOutgoing(context);
}
else {
this.handleIncoming(context);
}
return true;
}
@Override
public final boolean handleFault(SOAPMessageContext context) {
// Must pipe out Soap envelope
SOAPMessage soapMessage = context.getMessage();
TraceeContextLogger
.create()
.enforceOrder()
.apply()
.logWithPrefixedMessage(MessagePrefixProvider.provideLogMessagePrefix(MessageLogLevel.ERROR, "JAX-WS"), ImplicitContext.COMMON,
ImplicitContext.TRACEE, JaxWsWrapper.wrap(THREAD_LOCAL_SOAP_MESSAGE_STR.get(), convertSoapMessageAsString(soapMessage)));
return true;
}
@Override
public void close(MessageContext context) {
// cleanup thread local request soap message
THREAD_LOCAL_SOAP_MESSAGE_STR.remove();
}
@Override
public final Set getHeaders() {
return null;
}
/**
* Converts a SOAPMessage instance to string representation.
*/
String convertSoapMessageAsString(SOAPMessage soapMessage) {
if (soapMessage == null) {
return "null";
}
try {
ByteArrayOutputStream os = new ByteArrayOutputStream();
soapMessage.writeTo(os);
return new String(os.toByteArray(), determineMessageEncoding(soapMessage));
}
catch (Exception e) {
logger.error("Couldn't create string representation of soapMessage: " + soapMessage.toString());
return "ERROR";
}
}
Charset determineMessageEncoding(SOAPMessage soapMessage) {
try {
final Object encProp = soapMessage.getProperty(SOAPMessage.CHARACTER_SET_ENCODING);
if (encProp != null) {
return Charset.forName(String.valueOf(encProp));
}
return Charset.forName("UTF-8");
}
catch (Exception e) {
return Charset.forName("UTF-8");
}
}
protected void storeMessageInThreadLocal(SOAPMessageContext context) {
// Save soap request message in thread local storage for error logging
SOAPMessage soapMessage = context.getMessage();
if (soapMessage != null) {
String soapMessageAsString = convertSoapMessageAsString(soapMessage);
THREAD_LOCAL_SOAP_MESSAGE_STR.set(soapMessageAsString);
}
}
protected abstract void handleIncoming(SOAPMessageContext context);
protected abstract void handleOutgoing(SOAPMessageContext context);
/**
* Checks whether is is an incoming or outgoing call.
*
* @param messageContext the message context
* @return true if is outgoing call, otherwise false
*/
private boolean isOutgoing(MessageContext messageContext) {
Object outboundBoolean = messageContext.get(MessageContext.MESSAGE_OUTBOUND_PROPERTY);
return outboundBoolean != null && (Boolean)outboundBoolean;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy