All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.tentackle.persist.rmi.RemoteDelegateInvocationHandler Maven / Gradle / Ivy

There is a newer version: 21.16.1.0
Show newest version
/**
 * Tentackle - http://www.tentackle.org
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */


package org.tentackle.persist.rmi;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.rmi.RemoteException;
import java.util.Collection;
import java.util.Date;
import java.util.regex.Pattern;
import org.tentackle.common.LocaleProvider;
import org.tentackle.log.Loggable;
import org.tentackle.log.Logger;
import org.tentackle.log.Logger.Level;
import org.tentackle.log.LoggerFactory;
import org.tentackle.log.MappedDiagnosticContext;
import org.tentackle.misc.Duration;
import org.tentackle.misc.StringHelper;
import org.tentackle.pdo.Session;
import org.tentackle.persist.Db;
import org.tentackle.reflect.ReflectionHelper;


/**
 * Invocation handler to capture RMI method invocations.
 *
 * @author harald
 */
public class RemoteDelegateInvocationHandler implements InvocationHandler {

  /**
   * the mdc session key
   */
  public static final String MDC_SESSION_KEY = "session";

  /**
   * If true method invocations are counted and measured per method.
   * They can be logged via RemoteDbSession#logStatistics by the client request.
   * For debugging and profiling purposes only.
   */
  public static boolean collectStatistics;

  /**
   * If true, all invocations are logged.
   */
  public boolean logInvocations;

  /**
   * If true, args and return values will be logged as well.
   */
  public boolean logInvocationDetails;

  /**
   * Minimum milliseconds a method needs to execute before being logged.
* 0 to disable.

* For analysis and debugging purposes only. */ public static long logMinDurationMillis; /** * Minimum size of returned collection to get logged.
* 0 to disable.

* For analysis and debugging purposes only. */ public static int logMinReturnedCollectionSize; /** * Optional filter to log only RMI-calls for certain MappedDiagnosticContexts.
* If set, statements are only collected for logging if the MDC is valid and the pattern matches the toString-value * of the MDC. */ public static Pattern mdcFilter; /** * RMI thread name lead string. */ private static final String RMI_THREAD_NAME = "RMI TCP Connection"; /** * logger for this class. */ private static final Logger LOGGER = LoggerFactory.getLogger(RemoteDelegateInvocationHandler.class); /** * The delegate. */ private final RemoteDelegateImpl delegate; /** * Creates an invocation handler for a {@link RemoteDelegate}. *

* The invocation handler may be used to intercept remote method invocations. * The default implementation just sets the session info into the {@link MappedDiagnosticContext}. * * @param delegate the remote delegate */ public RemoteDelegateInvocationHandler(RemoteDelegateImpl delegate) { this.delegate = delegate; } /** * Gets the delegate. * * @return the delegate */ public RemoteDelegateImpl getDelegate() { return delegate; } /** * Updates the {@link MappedDiagnosticContext} with the session data. */ protected void updateMDC() { MappedDiagnosticContext mdc = LOGGER.getMappedDiagnosticContext(); RemoteDbSessionImpl session = delegate.getServerSession(); mdc.put(MDC_SESSION_KEY, session.getSessionNumber() + ":" + session.getClientSessionInfo().getUserName()); } /** * Clears the MDC. */ protected void clearMDC() { MappedDiagnosticContext mdc = LOGGER.getMappedDiagnosticContext(); mdc.remove(MDC_SESSION_KEY); } /** * Checks whether the mapped diagnostic context is valid for statistics logging. * * @return true if loggable */ protected boolean isMDCValid() { if (mdcFilter == null) { return true; } MappedDiagnosticContext mdc = LOGGER.getMappedDiagnosticContext(); return mdc != null && mdc.matchesPattern(mdcFilter); } /** * Updates the thread local memory. */ protected void updateThreadLocal() { delegate.getSession().makeCurrent(); LocaleProvider.getInstance().setCurrentLocale(delegate.getSession().getSessionInfo().getLocale()); } /** * Locks the Db for exclusive use by current thread. */ protected void lockDbForThread() { delegate.getSession().setOwnerThread(Thread.currentThread()); } /** * Handles an invocation exception.
* Figures out the real exception cause and throws it, * skipping InvocationTargetExceptions. * Furthermore, the exception will be logged in the server's log as well (not just the client). * * @param proxy the dynamic proxy * @param method the method to invoke * @param args the method's arguments * @param cause the invocation exception * @return the real RemoteException */ protected RemoteException handleInvocationException(Object proxy, Method method, Object[] args, Throwable cause) { // skip all InvocationTargetExceptions (could be chained if method itself is "proxied") while (cause instanceof InvocationTargetException) { Throwable nextCause = cause.getCause(); if (nextCause == null) { break; } cause = nextCause; } Level loggingLevel = Level.SEVERE; boolean withStacktrace = true; Throwable realCause = cause; try { if (cause instanceof RemoteException) { /** * log the exception on the server as well. * We use the info level because this exception * is meant for the client. */ realCause = cause.getCause(); if (realCause instanceof Loggable) { Loggable loggable = (Loggable) realCause; loggingLevel = loggable.getLogLevel(); withStacktrace = loggable.withStacktrace(); } else { loggingLevel = Level.INFO; } return (RemoteException) cause; } else { return new RemoteException("remote method execution failed", cause); } } finally { Db db = delegate.getSession(); boolean crashed = db != null && db.isCrashed(); if (crashed) { loggingLevel = Level.WARNING; } if (loggingLevel != null) { StringBuilder message = new StringBuilder(); message.append("delegate method ").append(method).append(" failed"); if (crashed) { message.append(" due to session cleanup"); } message.append(":\n").append(realCause.getMessage()); LOGGER.log(loggingLevel, message.toString(), withStacktrace ? realCause : null); } cleanupDbAfterException(); } } /** * Cleans up the db after an exception. *

* Invoked from {@link #handleInvocationException}. */ protected void cleanupDbAfterException() { Db db = delegate.getSession(); if (db != null && !db.isCrashed()) { try { // rollback any pending transaction and/or force detaching the db db.rollbackImmediately(); } catch (RuntimeException ex) { LOGGER.severe("*** immediate rollback failed ***", ex); } } } /** * Invokes the delegate's method. *

* If the invocation failed for some reason the stacktrace * is logged. * * @param proxy the dynamic proxy * @param method the method to invoke * @param args the method's arguments * @return the method's return value * @throws RemoteException if invocation failed */ protected Object invokeImpl(Object proxy, Method method, Object[] args) throws RemoteException { try { return method.invoke(delegate, args); } catch (Exception cause) { throw handleInvocationException(proxy, method, args, cause); } } /** * Returns whether measuring the duration is necessary. * * @return true if duration necessary */ protected boolean needDuration() { return logInvocations || collectStatistics || logMinDurationMillis > 0 || logMinReturnedCollectionSize > 0; } /** * Logs the invocation. *

* * @param method the method executed * @param args the method args * @param retval the method's return value * @param duration the duration when execution was started, null if don't care */ protected void log(Method method, Object[] args, Object retval, Duration duration) { boolean durationExceeded = false; if (duration != null) { if (!duration.isValid()) { duration.end(); } if (logMinDurationMillis > 0 && duration.millis() > logMinDurationMillis) { durationExceeded = true; } if (collectStatistics && isMDCValid()) { delegate.getServerSession().countMethodInvocation(method, delegate.getServicedClass(), duration); } } if (isMDCValid()) { int collectionSize = retval instanceof Collection ? ((Collection) retval).size() : 0; boolean collectionSizeExceeded = logMinReturnedCollectionSize > 0 && collectionSize > logMinReturnedCollectionSize; if (logInvocations || durationExceeded || collectionSizeExceeded) { StringBuilder buf = new StringBuilder(); if (logInvocationDetails) { buf.append(ReflectionHelper.getClassBaseName(delegate.getServicedClass())). append(": "). append(ReflectionHelper.getClassBaseName(delegate.getClass())). append('.'). append(method.getName()). append('('); if (args != null) { for (int i=0; i < args.length; i++) { if (i > 0) { buf.append(','); } buf.append(StringHelper.objectToLoggableString(args[i])); } } buf.append(") = "). append(StringHelper.objectToLoggableString(retval)). append(" {"). append(duration != null ? duration.millisToString() : ""). append("ms"); if (collectionSize > 0) { buf.append(", "). append(collectionSize). append(" items"); } buf.append('}'); } else { buf.append(delegate.getClass().getName()). append('.'). append(method.getName()). append(" ["). append(duration != null ? duration.millisToString() : ""). append("ms]"); } if (durationExceeded || collectionSizeExceeded) { if (durationExceeded) { buf.insert(0, ">>> duration >>> "); } if (collectionSizeExceeded) { buf.insert(0, ">>> size >>> "); } LOGGER.warning(buf.toString()); } else { LOGGER.info(buf.toString()); } } } // else no logging } /** * {@inheritDoc} *

* The default implementation just updates the MDC, invokes the method and logs the invoked method (args and return value) */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { RemoteDbSessionImpl serverSession = delegate.getServerSession(); /** * Because RMI requests may be executed by different threads (from the RMI's threadpool) * we must make sure that successive invocations using the same session (mapped 1:1 to the Db instances) * see each other's modifications. Hence, we synchronize on the session. * Because no two RMI-requests refer to the same session simultaneously (except the client * is misbehaving) the object monitor on the session will never cause a deadlock because * of another conflicting synchronization. */ synchronized (serverSession) { if (!serverSession.isOpen()) { throw new RemoteSessionClosedException(serverSession + " already closed since " + new Date(serverSession.getClosedSince())); } InvocationLocals locals = Thread.currentThread().getName().startsWith(RMI_THREAD_NAME) ? new InvocationLocals(delegate, LOGGER.getMappedDiagnosticContext(), MDC_SESSION_KEY) : null; if (locals != null) { // mark session still alive delegate.getSession().setAlive(true); try { // update thread local memory updateThreadLocal(); // set the MDC for logging updateMDC(); // invoke the method with Db temporarily locked to the current thread lockDbForThread(); } catch (Throwable t) { // for some weird reasons the session is in use or whatever. // Can't happen, but if it does, we need to fix this somehow. LOGGER.severe("cleanup server session locals after exception", t); delegate.getSession().setOwnerThread(null); Session.setCurrentSession(null); LocaleProvider.getInstance().setCurrentLocale(null); clearMDC(); throw t; } } try { Duration duration = needDuration() ? new Duration() : null; Object retval = invokeImpl(proxy, method, args); log(method, args, retval, duration); return retval; } finally { if (locals != null) { locals.restore(); } } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy