001/**
002 * Copyright 2010 The Kuali Foundation Licensed under the
003 * Educational Community License, Version 2.0 (the "License"); you may
004 * not use this file except in compliance with the License. You may
005 * obtain a copy of the License at
006 *
007 * http://www.osedu.org/licenses/ECL-2.0
008 *
009 * Unless required by applicable law or agreed to in writing,
010 * software distributed under the License is distributed on an "AS IS"
011 * BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
012 * or implied. See the License for the specific language governing
013 * permissions and limitations under the License.
014 */
015
016package org.kuali.student.common.util;
017
018import java.io.PrintWriter;
019import java.io.StringWriter;
020import java.lang.reflect.Constructor;
021import java.util.Map;
022
023import org.apache.log4j.Logger;
024import org.springframework.aop.ThrowsAdvice;
025import org.springframework.core.Ordered;
026
027/**
028 * @author Daniel Epstein
029 *         <p>
030 *         Use this Advice to map one exception to another for use when other
031 *         advice eats your runtime exceptions outside of your code. This happens in
032 *         Transactions when commit is not called until outside of your DAO
033 *         layer.
034 *         </p>
035 * 
036 * <p>
037 * Set the property "exceptionMapping" as a map that maps an exception class to
038 * your own exception class
039 * </p>
040 * 
041 * <p>
042 * Remember that aspect order is important and that this bean will always be
043 * order "500"
044 * </p>
045 * 
046 * Example:
047 * 
048 * <pre>
049 * &lt;tx:annotation-driven transaction-manager=&quot;JtaTxManager&quot; order=&quot;1000&quot;/&gt;
050 * lt;bean id=&quot;mapExceptionAdvisor&quot;
051 * class=&quot;org.myfoo.ExceptionMappingAdvice&quot;&gt;
052 * &lt;property name=&quot;exceptionMapping&quot;&gt;
053 *      &lt;map&gt;
054 *              &lt;entry key=&quot;javax.persistence.EntityExistsException&quot;
055 *                      value=&quot;org.myfoo.exceptions.AlreadyExistsException&quot; /&gt;
056 *      &lt;/map&gt;
057 * &lt;/property&gt;
058 * lt;/bean&gt;
059 * lt;aop:config&gt;
060 * &lt;aop:aspect id=&quot;dataAccessToBusinessException&quot;
061 *      ref=&quot;mapExceptionAdvisor&quot;&gt;
062 *      &lt;aop:after-throwing
063 *              pointcut=&quot;execution(* org.myfoo.service.*.*(..))&quot;
064 *              method=&quot;afterThrowing&quot; throwing=&quot;ex&quot; /&gt;
065 * &lt;/aop:aspect&gt;
066 * lt;/aop:config&gt;
067 * </pre>
068 */
069public class ExceptionMappingAdvice implements ThrowsAdvice, Ordered {
070        private int order = 500;
071        private static final long serialVersionUID = 1L;
072        private Map<Class<? extends Exception>, Class<? extends Exception>> exceptionMapping;
073        private Class<? extends Exception> defaultException;
074        final Logger logger = Logger.getLogger(ExceptionMappingAdvice.class);
075
076        /**
077         * This method will use the real exception thrown and look up the exception
078         * that should be thrown
079         * 
080         * @param ex
081         * @throws Exception
082         */
083        public void afterThrowing(Exception ex) throws Exception {
084                Class<? extends Exception> mappedExceptionClass = exceptionMapping
085                                .get(ex.getClass());
086
087                if (mappedExceptionClass != null) {
088            if (logger.isDebugEnabled()) {
089                logger.debug(String.format("Mapping exception %s to %s", ex.getClass(), mappedExceptionClass));
090            }
091            Constructor<? extends Exception> c = mappedExceptionClass
092                                        .getConstructor(String.class);
093            throw c.newInstance(ex.getMessage());
094                }
095                
096                //Throw a default exception if this is a runtime exception
097                if(ex instanceof RuntimeException){
098                        logger.trace("No mapping available, throwing default exception "+defaultException);
099                        if (defaultException != null) {
100                                //Log the error
101                                StringWriter traceWriter = new StringWriter();
102                                PrintWriter printWriter = new PrintWriter(traceWriter, false);
103                                logger.error(printWriter, ex);
104                                printWriter.close();
105                                String faultMessage = traceWriter.getBuffer().toString();
106                                logger.error(faultMessage);
107                                //Throw the default exception
108                                try{
109                                        Constructor<? extends Exception> c = defaultException
110                                                        .getConstructor(String.class, Throwable.class);
111                                        throw c.newInstance(ex.getMessage(), ex);
112                                }catch(NoSuchMethodException e){
113                                        Constructor<? extends Exception> c = defaultException
114                                                        .getConstructor(String.class);
115                                        throw c.newInstance(ex.getMessage());
116                                }
117                        }
118                        //Check if no default was defined
119            if (logger.isDebugEnabled()) {
120                logger.debug(String.format("No mapping or default exception available. Exception %s", ex.getClass()));
121            }
122            throw new RuntimeException("Could Not Map Exception: " + ex.toString());
123                }
124        }
125
126        @Override
127        public int getOrder() {
128                return order;
129        }
130
131        /**
132         * @param order
133         *            the order to set
134         */
135        public void setOrder(int order) {
136                this.order = order;
137        }
138
139        /**
140         * @return the exceptionMapping
141         */
142        public Map<Class<? extends Exception>, Class<? extends Exception>> getExceptionMapping() {
143                return exceptionMapping;
144        }
145
146        /**
147         * @param exceptionMapping
148         *            the exceptionMapping to set
149         */
150        public void setExceptionMapping(
151                        Map<Class<? extends Exception>, Class<? extends Exception>> exceptionMapping) {
152                this.exceptionMapping = exceptionMapping;
153        }
154
155        /**
156         * @return the defaultException
157         */
158        public Class<? extends Exception> getDefaultException() {
159                return defaultException;
160        }
161
162        /**
163         * @param defaultException
164         *            the defaultException to set
165         */
166        public void setDefaultException(Class<? extends Exception> defaultException) {
167                this.defaultException = defaultException;
168        }
169
170}