001 /** 002 * Copyright 2005-2011 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.krad.util.spring; 017 018 import java.lang.reflect.Method; 019 020 import org.aopalliance.intercept.MethodInterceptor; 021 import org.aopalliance.intercept.MethodInvocation; 022 import org.apache.commons.lang.StringUtils; 023 import org.apache.commons.logging.Log; 024 import org.apache.commons.logging.LogFactory; 025 026 /** 027 * This interceptor generates log message before entering and after leaving the method being intercepted. 028 * 029 * 030 */ 031 public class MethodLoggingInterceptor implements MethodInterceptor { 032 private static final Log LOG = LogFactory.getLog(MethodLoggingInterceptor.class); 033 034 035 /** 036 * Surrounds the method invocation with FATAL-level log messages. Using FATAL because I want to make sure that the log messages 037 * will always show up, since the correct way to deactivate method-level logging is to remove (or comment) the bean name from 038 * the KualiTestSpringLogging.xml file. 039 * 040 * @see org.aopalliance.intercept.MethodInterceptor#invoke(org.aopalliance.intercept.MethodInvocation) 041 */ 042 public Object invoke(MethodInvocation invocation) throws Throwable { 043 long startTime = System.currentTimeMillis(); 044 Object methodResult = null; 045 String invocationLabel = buildInvocationLabel(invocation); 046 try { 047 LOG.fatal("entering " + invocationLabel); 048 049 methodResult = invocation.proceed(); 050 } 051 catch (Exception invocationException) { 052 String exceptionLabel = buildExceptionLabel(invocationException); 053 LOG.fatal("aborting " + invocationLabel + ": throwing " + exceptionLabel); 054 055 throw invocationException; 056 } 057 LOG.fatal(new StringBuffer("leaving ").append(invocationLabel).append(" / took ").append(System.currentTimeMillis() - startTime).append(" ms")); 058 059 return methodResult; 060 } 061 062 /** 063 * @param invocation MethodInvocation being labeled 064 * @return String used to identify this MethodInvocation 065 */ 066 private String buildInvocationLabel(MethodInvocation invocation) { 067 StringBuffer invocationLabel = new StringBuffer(); 068 069 Method method = invocation.getMethod(); 070 Class targetClass = invocation.getThis().getClass(); 071 Class declaringClass = method.getDeclaringClass(); 072 073 // {targetClass} declaringClass.method 074 if (targetClass != declaringClass) { 075 invocationLabel.append("{"); 076 invocationLabel.append(declaringClass.getName()); 077 invocationLabel.append("} "); 078 } 079 invocationLabel.append(targetClass.getName() + "." + method.getName()); 080 081 082 // (paramClass=argValue[,paramClass=argValue...]) 083 Class[] paramTypes = method.getParameterTypes(); 084 Object[] argValues = invocation.getArguments(); 085 086 invocationLabel.append("("); 087 if (paramTypes != null) { 088 for (int i = 0; i < paramTypes.length; i++) { 089 if (i > 0) { 090 invocationLabel.append(","); 091 } 092 093 invocationLabel.append(paramTypes[i].getName()); 094 invocationLabel.append("="); 095 096 // differentiate between literal null and object whose toString returns null 097 if (argValues[i] == null) { 098 invocationLabel.append("<literal null>"); 099 } 100 else { 101 invocationLabel.append(argValues[i]); 102 } 103 } 104 } 105 invocationLabel.append(")"); 106 107 return invocationLabel.toString(); 108 } 109 110 111 /** 112 * @param e Exception being labeled 113 * @return String used to identify this Exception 114 */ 115 private String buildExceptionLabel(Exception e) { 116 String className = e.getClass().getName(); 117 118 String exceptionLabel = StringUtils.substringAfterLast(className, "."); 119 if (StringUtils.isBlank(exceptionLabel)) { 120 exceptionLabel = className; 121 } 122 123 return exceptionLabel; 124 } 125 }