View Javadoc

1   /**
2    * Copyright 2005-2012 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.kns.web.struts.action;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.apache.log4j.Logger;
20  import org.apache.log4j.MDC;
21  import org.apache.ojb.broker.OptimisticLockException;
22  import org.apache.struts.Globals;
23  import org.apache.struts.action.Action;
24  import org.apache.struts.action.ActionForm;
25  import org.apache.struts.action.ActionForward;
26  import org.apache.struts.action.ActionMapping;
27  import org.apache.struts.action.InvalidCancelException;
28  import org.apache.struts.action.RequestProcessor;
29  import org.apache.struts.config.FormBeanConfig;
30  import org.apache.struts.config.ForwardConfig;
31  import org.apache.struts.util.RequestUtils;
32  import org.kuali.rice.core.api.util.RiceConstants;
33  import org.kuali.rice.core.api.util.RiceKeyConstants;
34  import org.kuali.rice.kns.exception.FileUploadLimitExceededException;
35  import org.kuali.rice.kns.service.KNSServiceLocator;
36  import org.kuali.rice.kns.service.SessionDocumentService;
37  import org.kuali.rice.kns.util.ErrorContainer;
38  import org.kuali.rice.kns.util.InfoContainer;
39  import org.kuali.rice.kns.util.KNSConstants;
40  import org.kuali.rice.kns.util.KNSGlobalVariables;
41  import org.kuali.rice.kns.util.WarningContainer;
42  import org.kuali.rice.kns.util.WebUtils;
43  import org.kuali.rice.kns.web.EditablePropertiesHistoryHolder;
44  import org.kuali.rice.kns.web.struts.form.KualiDocumentFormBase;
45  import org.kuali.rice.kns.web.struts.form.KualiForm;
46  import org.kuali.rice.kns.web.struts.form.pojo.PojoForm;
47  import org.kuali.rice.krad.UserSession;
48  import org.kuali.rice.krad.document.Document;
49  import org.kuali.rice.krad.exception.ValidationException;
50  import org.kuali.rice.krad.service.KRADServiceLocatorInternal;
51  import org.kuali.rice.krad.util.GlobalVariables;
52  import org.kuali.rice.krad.util.KRADConstants;
53  import org.kuali.rice.krad.util.KRADUtils;
54  import org.kuali.rice.krad.util.MessageMap;
55  import org.springframework.transaction.PlatformTransactionManager;
56  import org.springframework.transaction.TransactionStatus;
57  import org.springframework.transaction.support.TransactionCallback;
58  import org.springframework.transaction.support.TransactionTemplate;
59  import org.springmodules.orm.ojb.OjbOperationException;
60  
61  import javax.servlet.ServletException;
62  import javax.servlet.http.HttpServletRequest;
63  import javax.servlet.http.HttpServletResponse;
64  import javax.servlet.http.HttpSession;
65  import java.io.IOException;
66  
67  /**
68   * This class handles setup of user session and restoring of action form.
69   * 
70   * 
71   */
72  public class KualiRequestProcessor extends RequestProcessor {
73  	
74  	private static final String MDC_DOC_ID = "docId";
75  	private static final String PREVIOUS_REQUEST_EDITABLE_PROPERTIES_GUID_PARAMETER_NAME = "actionEditablePropertiesGuid";
76  
77  	private static Logger LOG = Logger.getLogger(KualiRequestProcessor.class);
78  
79  	private SessionDocumentService sessionDocumentService;
80  	private PlatformTransactionManager transactionManager;
81  	
82  	@Override
83  	public void process(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
84  		if ( LOG.isInfoEnabled() ) {
85  			LOG.info(new StringBuffer("Started processing request: '").append(request.getRequestURI()).append("' w/ query string: '").append(request.getQueryString()).append("'"));
86  		}
87  
88  		try { 
89  			strutsProcess(request, response);
90  		} catch (FileUploadLimitExceededException e) {
91  			ActionForward actionForward = processException(request, response, e, e.getActionForm(), e.getActionMapping());
92  			processForwardConfig(request, response, actionForward);
93  		} finally {
94  			KNSGlobalVariables.setKualiForm(null);
95  		}
96  			
97  		try {
98  			ActionForm form = WebUtils.getKualiForm(request);
99  			
100 			if (form != null && form instanceof KualiDocumentFormBase) {
101 				String docId = ((KualiDocumentFormBase) form).getDocId();
102 				if (docId != null) { MDC.put(MDC_DOC_ID, docId); }
103 			}
104 
105 			String refreshCaller = request.getParameter(KRADConstants.REFRESH_CALLER);
106 			if (form!=null && KualiDocumentFormBase.class.isAssignableFrom(form.getClass()) 
107 					&& !KRADConstants.QUESTION_REFRESH.equalsIgnoreCase(refreshCaller)) {
108 				KualiDocumentFormBase docForm = (KualiDocumentFormBase) form;
109 				Document document = docForm.getDocument();
110 				String docFormKey = docForm.getFormKey();
111 
112 				UserSession userSession = (UserSession) request.getSession().getAttribute(KRADConstants.USER_SESSION_KEY);
113 
114 				if (WebUtils.isDocumentSession(document, docForm)) {
115 					getSessionDocumentService().setDocumentForm(docForm, userSession, request.getRemoteAddr());
116 				}
117 
118 				Boolean exitingDocument = (Boolean) request.getAttribute(KRADConstants.EXITING_DOCUMENT);
119 
120 				if (exitingDocument != null && exitingDocument.booleanValue()) {
121 					// remove KualiDocumentFormBase object from session and
122 					// table.
123 					getSessionDocumentService().purgeDocumentForm(docForm.getDocument().getDocumentNumber(), docFormKey, userSession, request.getRemoteAddr());
124 				}
125 			}
126 
127 			if ( LOG.isInfoEnabled() ) {
128 				LOG.info(new StringBuffer("Finished processing request: '").append(request.getRequestURI()).append("' w/ query string: '").append(request.getQueryString()).append("'"));
129 			}
130 
131 		} finally {
132 			// MDC docId key is set above, and also during super.process() in the call to processActionForm
133 			MDC.remove(MDC_DOC_ID);
134 		}
135 
136 	}
137 	
138 	@Override
139 	protected boolean processPreprocess(HttpServletRequest request, HttpServletResponse response) {
140         final UserSession session = KRADUtils.getUserSessionFromRequest(request);
141         
142         if (session == null) {
143         	throw new IllegalStateException("the user session has not been established");
144         }
145     	GlobalVariables.setUserSession(session);
146     	KNSGlobalVariables.clear();
147 		return true;
148 	}
149 	
150 	/**
151      * <p>ProcessDefinition an <code>HttpServletRequest</code> and create the
152      * corresponding <code>HttpServletResponse</code> or dispatch
153      * to another resource.</p>
154      *
155      * @param request The servlet request we are processing
156      * @param response The servlet response we are creating
157      *
158      * @exception IOException if an input/output error occurs
159      * @exception ServletException if a processing exception occurs
160      */
161     public void strutsProcess(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
162 
163         // Wrap multipart requests with a special wrapper
164         request = processMultipart(request);
165 
166         // Identify the path component we will use to select a mapping
167         String path = processPath(request, response);
168         if (path == null) {
169             return;
170         }
171         
172         if (log.isDebugEnabled()) {
173             log.debug("Processing a '" + request.getMethod() +
174                       "' for path '" + path + "'");
175         }
176 
177         // Select a Locale for the current user if requested
178         processLocale(request, response);
179 
180         // Set the content type and no-caching headers if requested
181         processContent(request, response);
182         processNoCache(request, response);
183 
184         // General purpose preprocessing hook
185         if (!processPreprocess(request, response)) {
186             return;
187         }
188         
189         this.processCachedMessages(request, response);
190 
191         // Identify the mapping for this request
192         ActionMapping mapping = processMapping(request, response, path);
193         if (mapping == null) {
194             return;
195         }
196 
197         // Check for any role required to perform this action
198         if (!processRoles(request, response, mapping)) {
199             return;
200         }
201 
202         processFormActionAndForward(request, response, mapping);
203 
204     }
205 
206     public void processFormActionAndForward(final HttpServletRequest request, final HttpServletResponse response, final ActionMapping mapping) throws ServletException, IOException {
207     	ActionForm form = processActionForm(request, response, mapping);
208         processPopulate(request, response, form, mapping);
209 
210         // Create or acquire the Action instance to process this request
211 		Action action = processActionCreate(request, response, mapping);
212 
213         if (action != null) {
214             // Call the Action instance itself
215 		    ActionForward forward = processActionPerform(request, response, action, form, mapping);
216 
217             if (forward != null) {
218                 // ProcessDefinition the returned ActionForward instance
219 			    processForwardConfig(request, response, forward);
220             }
221         }
222     }
223 
224 
225 	/**
226 	 * This method gets the document number from the request.  The request should have been processed already 
227 	 * before this is called if it is multipart.  
228 	 * 
229 	 * @param request
230 	 * @return the document number, or null if one can't be found in the request.
231 	 */
232 	private String getDocumentNumber(HttpServletRequest request) {
233 		String documentNumber = request.getParameter(KRADConstants.DOCUMENT_DOCUMENT_NUMBER);
234 
235 		// from lookup pages.
236 		if (documentNumber == null) {
237 			documentNumber = request.getParameter(KRADConstants.DOC_NUM);
238 		}
239 		
240 		if (documentNumber == null) {
241 			documentNumber = request.getParameter("documentId");
242 		}
243 		
244 		return documentNumber;
245 	}
246 
247 	/**
248 	 * Hooks into populate process to call form populate method if form is an
249 	 * instanceof PojoForm.
250 	 */
251 	@Override
252 	protected void processPopulate(HttpServletRequest request, HttpServletResponse response, ActionForm form, ActionMapping mapping) throws ServletException {
253 		if (form instanceof KualiForm) {
254 			// Add the ActionForm to GlobalVariables
255 			// This will allow developers to retrieve both the Document and any
256 			// request parameters that are not
257 			// part of the Form and make them available in ValueFinder classes
258 			// and other places where they are needed.
259 			KNSGlobalVariables.setKualiForm((KualiForm) form);
260 		}
261 
262 		// if not PojoForm, call struts populate
263 		if (!(form instanceof PojoForm)) {
264 			super.processPopulate(request, response, form, mapping);
265 			return;
266 		}
267 		
268 		final String previousRequestGuid = request.getParameter(KualiRequestProcessor.PREVIOUS_REQUEST_EDITABLE_PROPERTIES_GUID_PARAMETER_NAME);
269 
270 		((PojoForm)form).clearEditablePropertyInformation();
271 		((PojoForm)form).registerStrutsActionMappingScope(mapping.getScope());
272 		
273 		String multipart = mapping.getMultipartClass();
274 		if (multipart != null) {
275 			request.setAttribute(Globals.MULTIPART_KEY, multipart);
276 		}
277 
278 		form.setServlet(this.servlet);
279 		form.reset(mapping, request);
280 
281 		((PojoForm)form).setPopulateEditablePropertiesGuid(previousRequestGuid);
282 		// call populate on ActionForm
283 		((PojoForm) form).populate(request);
284 		request.setAttribute("UnconvertedValues", ((PojoForm) form).getUnconvertedValues().keySet());
285 		request.setAttribute("UnconvertedHash", ((PojoForm) form).getUnconvertedValues());
286 	}
287 
288 	/**
289 	 * Hooks into validate to catch any errors from the populate, and translate
290 	 * the ErrorMap to ActionMessages.
291 	 */
292 	@Override
293 	protected boolean processValidate(HttpServletRequest request, HttpServletResponse response, ActionForm form, ActionMapping mapping) throws IOException, ServletException, InvalidCancelException {
294 
295 		// skip form validate if we had errors from populate
296 		if (GlobalVariables.getMessageMap().hasNoErrors()) {
297 			if (form == null) {
298 				return (true);
299 			}
300 			// Was this request cancelled?
301 			if (request.getAttribute(Globals.CANCEL_KEY) != null) {
302 				if (LOG.isDebugEnabled()) {
303 					LOG.debug(" Cancelled transaction, skipping validation");
304 				}
305 				return (true);
306 			}
307 
308 			// Has validation been turned off for this mapping?
309 			if (!mapping.getValidate()) {
310 				return (true);
311 			}
312 
313 			// call super to call forms validate
314 			super.processValidate(request, response, form, mapping);
315 		}
316 
317 		publishMessages(request);
318 		if (!GlobalVariables.getMessageMap().hasNoErrors()) {
319 			// Special handling for multipart request
320 			if (form.getMultipartRequestHandler() != null) {
321 				if (LOG.isDebugEnabled()) {
322 					LOG.debug("  Rolling back multipart request");
323 				}
324 				form.getMultipartRequestHandler().rollback();
325 			}
326 
327 			// Fix state that could be incorrect because of validation failure
328 			if (form instanceof PojoForm) {
329 				((PojoForm) form).processValidationFail();
330 			}
331 
332 			// Was an input path (or forward) specified for this mapping?
333 			String input = mapping.getInput();
334 			if (input == null) {
335 				if (LOG.isDebugEnabled()) {
336 					LOG.debug("  Validation failed but no input form available");
337 				}
338 				response.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR, getInternal().getMessage("noInput", mapping.getPath()));
339 				return (false);
340 			}
341 
342 			if (moduleConfig.getControllerConfig().getInputForward()) {
343 				ForwardConfig forward = mapping.findForward(input);
344 				processForwardConfig(request, response, forward);
345 			} else {
346 				internalModuleRelativeForward(input, request, response);
347 			}
348 
349 			return (false);
350 		}
351 		return true;
352 	}
353 
354 	/**
355 	 * Checks for return from a lookup or question, and restores the action form
356 	 * stored under the request parameter docFormKey.
357 	 */
358 	@Override
359 	protected ActionForm processActionForm(HttpServletRequest request, HttpServletResponse response, ActionMapping mapping) {
360 		
361 		String documentNumber = getDocumentNumber(request);
362 		if (documentNumber != null) { MDC.put(MDC_DOC_ID, documentNumber); }
363 		
364 		UserSession userSession = (UserSession) request.getSession().getAttribute(KRADConstants.USER_SESSION_KEY);
365 
366 		String docFormKey = request.getParameter(KRADConstants.DOC_FORM_KEY);
367 		String methodToCall = request.getParameter(KRADConstants.DISPATCH_REQUEST_PARAMETER);
368 		String refreshCaller = request.getParameter(KRADConstants.REFRESH_CALLER);
369 //		String searchListRequestKey = request.getParameter(KRADConstants.SEARCH_LIST_REQUEST_KEY);
370 		String documentWebScope = request.getParameter(KRADConstants.DOCUMENT_WEB_SCOPE);
371 
372 		if (mapping.getPath().startsWith(KRADConstants.REFRESH_MAPPING_PREFIX) || KRADConstants.RETURN_METHOD_TO_CALL.equalsIgnoreCase(methodToCall) ||
373 				KRADConstants.QUESTION_REFRESH.equalsIgnoreCase(refreshCaller) || KRADConstants.TEXT_AREA_REFRESH.equalsIgnoreCase(refreshCaller) || KRADConstants
374                 .SESSION_SCOPE.equalsIgnoreCase(documentWebScope)) {
375 			ActionForm form = null;
376 			// check for search result storage and clear
377 			GlobalVariables.getUserSession().removeObjectsByPrefix(KRADConstants.SEARCH_LIST_KEY_PREFIX);
378 
379 			// We put different type of forms such as document form, lookup form
380 			// in session but we only store document form in
381 			// database.
382 			if (userSession.retrieveObject(docFormKey) != null) {
383 				LOG.debug("getDecomentForm KualiDocumentFormBase from session");
384 				form = (ActionForm) userSession.retrieveObject(docFormKey);
385 			} else if (StringUtils.isNotBlank(documentNumber)) {
386 				form = getSessionDocumentService().getDocumentForm(documentNumber, docFormKey, userSession, request.getRemoteAddr());
387 			}
388 			request.setAttribute(mapping.getAttribute(), form);
389 			if (!KRADConstants.SESSION_SCOPE.equalsIgnoreCase(documentWebScope)) {
390 				userSession.removeObject(docFormKey);
391 			}
392 			// we should check whether this is a multipart request because we
393 			// could have had a combination of query parameters and a multipart
394 			// request
395 			String contentType = request.getContentType();
396 			String method = request.getMethod();
397 			if (("POST".equalsIgnoreCase(method) && contentType != null && contentType.startsWith("multipart/form-data"))) {
398 				// this method parses the multipart request and adds new
399 				// non-file parameters into the request
400 				WebUtils.getMultipartParameters(request, null, form, mapping);
401 			}
402 			// The form can be null if the document is not a session document
403 			if (form != null) {
404 				return form;
405 			}
406 		}
407 
408 		// Rice has the ability to limit file upload sizes on a per-form basis,
409 		// so the max upload sizes may be accessed by calling methods on
410 		// PojoFormBase.
411 		// This requires that we are able know the file upload size limit (i.e.
412 		// retrieve a form instance) before we parse a mulitpart request.
413 		ActionForm form = super.processActionForm(request, response, mapping);
414 
415 		// for sessiondocument with multipart request
416 		String contentType = request.getContentType();
417 		String method = request.getMethod();
418 
419 		if ("GET".equalsIgnoreCase(method) && StringUtils.isNotBlank(methodToCall) && form instanceof PojoForm &&
420 				((PojoForm) form).getMethodToCallsToBypassSessionRetrievalForGETRequests().contains(methodToCall)) {
421 			return createNewActionForm(mapping, request);
422 		}
423 		
424 		// if we have a multipart request, parse it and return the stored form
425 		// from session if the doc form key is not blank. If it is blank, then
426 		// we just return the form
427 		// generated from the superclass processActionForm method. Either way,
428 		// we need to parse the mulitpart request now so that we may determine
429 		// what the value of the doc form key is.
430 		// This is generally against the contract of processActionForm, because
431 		// processPopulate should be responsible for parsing the mulitpart
432 		// request, but we need to parse it now
433 		// to determine the doc form key value.
434 		if (("POST".equalsIgnoreCase(method) && contentType != null && contentType.startsWith("multipart/form-data"))) {
435 			WebUtils.getMultipartParameters(request, null, form, mapping);
436 			docFormKey = request.getParameter(KRADConstants.DOC_FORM_KEY);
437 			documentWebScope = request.getParameter(KRADConstants.DOCUMENT_WEB_SCOPE);
438 
439 			documentNumber = getDocumentNumber(request);
440 
441 			if (KRADConstants.SESSION_SCOPE.equalsIgnoreCase(documentWebScope) ||
442 					(form instanceof KualiDocumentFormBase && WebUtils
443                             .isDocumentSession(((KualiDocumentFormBase) form).getDocument(),
444                                     (KualiDocumentFormBase) form))) {
445 
446 				Object userSessionObject = userSession.retrieveObject(docFormKey);
447 				if ( userSessionObject != null &&  userSessionObject instanceof ActionForm ) {
448 					LOG.debug("getDocumentForm KualiDocumentFormBase from session");
449 					form = (ActionForm) userSessionObject;
450 				} else {
451 					ActionForm tempForm = getSessionDocumentService().getDocumentForm(documentNumber, docFormKey, userSession, request.getRemoteAddr());
452 					if ( tempForm != null ) {
453 						form = tempForm;
454 					}
455 				}
456 
457 				request.setAttribute(mapping.getAttribute(), form);
458 				if (form != null) {
459 					return form;
460 				}
461 			}
462 		}
463 		return form;
464 	}
465 
466 	/**
467 	 * Hook into action perform to handle errors in the error map and catch
468 	 * exceptions.
469 	 * 
470 	 * <p>
471 	 * A transaction is started prior to the execution of the action. This
472 	 * allows for the action code to execute efficiently without the need for
473 	 * using PROPAGATION_SUPPORTS in the transaction definitions. The
474 	 * PROPAGATION_SUPPORTS propagation type does not work well with JTA.
475 	 */
476 	@Override
477 	protected ActionForward processActionPerform(final HttpServletRequest request, final HttpServletResponse response, final Action action, final ActionForm form, final ActionMapping mapping) throws IOException, ServletException {
478 		try {
479             TransactionTemplate template = new TransactionTemplate(getTransactionManager());
480 			ActionForward forward = null;
481 			try {
482 				forward = (ActionForward) template.execute(new TransactionCallback() {
483 					public Object doInTransaction(TransactionStatus status) {
484 						ActionForward actionForward = null;
485 						try {
486 							actionForward = action.execute(mapping, form, request, response);
487 						} catch (Exception e) {
488 							// the doInTransaction method has no means for
489 							// throwing exceptions, so we will wrap the
490 							// exception in
491 							// a RuntimeException and re-throw. The one caveat
492 							// here is that this will always result in
493 							// the
494 							// transaction being rolled back (since
495 							// WrappedRuntimeException is a runtime exception).
496 							throw new WrappedRuntimeException(e);
497 						}
498 						if (status.isRollbackOnly()) {
499 							// this means that the struts action execution
500 							// caused the transaction to rollback, we want to
501 							// go ahead
502 							// and trigger the rollback by throwing an exception
503 							// here but then return the action forward
504 							// from this method
505 							throw new WrappedActionForwardRuntimeException(actionForward);
506 						}
507 						return actionForward;
508 					}
509 				});
510 			} catch (WrappedActionForwardRuntimeException e) {
511 				forward = e.getActionForward();
512 			}
513 
514 			publishMessages(request);
515 			saveMessages(request);
516 			saveAuditErrors(request);
517 
518 			if (form instanceof PojoForm) {
519 				if (((PojoForm)form).getEditableProperties() == null
520 						|| ((PojoForm)form).getEditableProperties().isEmpty()) {
521 					EditablePropertiesHistoryHolder holder = (EditablePropertiesHistoryHolder) GlobalVariables.getUserSession().getObjectMap().get(
522                             KRADConstants.EDITABLE_PROPERTIES_HISTORY_HOLDER_ATTR_NAME);
523 				    if (holder == null) {
524 				    	holder = new EditablePropertiesHistoryHolder();
525 				    }
526 
527 					final String guid = holder.addEditablePropertiesToHistory(((PojoForm)form).getEditableProperties());
528 				    ((PojoForm)form).setActionEditablePropertiesGuid(guid);
529 				    GlobalVariables.getUserSession().addObject(KRADConstants.EDITABLE_PROPERTIES_HISTORY_HOLDER_ATTR_NAME, holder);
530 				}
531 			}
532 
533 			return forward;
534 
535 		} catch (Exception e) {
536 			if (e instanceof WrappedRuntimeException) {
537 				e = (Exception) e.getCause();
538 			}
539 			if (e instanceof ValidationException) {
540 				// add a generic error message if there are none
541 				if (GlobalVariables.getMessageMap().hasNoErrors()) {
542 
543 					GlobalVariables.getMessageMap().putError(KRADConstants.GLOBAL_ERRORS, RiceKeyConstants.ERROR_CUSTOM, e.getMessage());
544 				}
545 
546 				if (form instanceof PojoForm) {
547 					if (((PojoForm)form).getEditableProperties() == null
548 							|| ((PojoForm)form).getEditableProperties().isEmpty()) {
549 					    EditablePropertiesHistoryHolder holder = (EditablePropertiesHistoryHolder) GlobalVariables.getUserSession().getObjectMap().get(
550                                 KRADConstants.EDITABLE_PROPERTIES_HISTORY_HOLDER_ATTR_NAME);
551 				        if (holder == null) {
552 				    	    holder = new EditablePropertiesHistoryHolder();
553 				        }
554 
555 					    final String guid = holder.addEditablePropertiesToHistory(((PojoForm)form).getEditableProperties());
556 				        ((PojoForm)form).setActionEditablePropertiesGuid(guid);
557 				        GlobalVariables.getUserSession().addObject(KRADConstants.EDITABLE_PROPERTIES_HISTORY_HOLDER_ATTR_NAME, holder);
558 					}
559 				}			
560 				// display error messages and return to originating page
561 				publishMessages(request);
562 				return mapping.findForward(RiceConstants.MAPPING_BASIC);
563 			}
564 
565 			publishMessages(request);
566 
567 			return (processException(request, response, e, form, mapping));
568 		}
569 	}
570 
571     private static class WrappedActionForwardRuntimeException extends RuntimeException {
572 		private ActionForward actionForward;
573 
574 		public WrappedActionForwardRuntimeException(ActionForward actionForward) {
575 			this.actionForward = actionForward;
576 		}
577 
578 		public ActionForward getActionForward() {
579 			return actionForward;
580 		}
581 	}
582 
583 	/**
584 	 * Adds more detailed logging for unhandled exceptions
585 	 * 
586 	 * @see org.apache.struts.action.RequestProcessor#processException(HttpServletRequest,
587 	 *      HttpServletResponse, Exception, ActionForm, ActionMapping)
588 	 */
589 	@Override
590 	protected ActionForward processException(HttpServletRequest request, HttpServletResponse response, Exception exception, ActionForm form, ActionMapping mapping) throws IOException, ServletException {
591 		ActionForward actionForward = null;
592 
593 		try {
594 			actionForward = super.processException(request, response, exception, form, mapping);
595 		} catch (IOException e) {
596 			logException(e);
597 			throw e;
598 		} catch (ServletException e) {
599 			// special case, to make OptimisticLockExceptions easier to read
600 			Throwable rootCause = e.getRootCause();
601 			if (rootCause instanceof OjbOperationException) {
602 				OjbOperationException ooe = (OjbOperationException) rootCause;
603 
604 				Throwable subcause = ooe.getCause();
605 				if (subcause instanceof OptimisticLockException) {
606 					OptimisticLockException ole = (OptimisticLockException) subcause;
607 
608 					StringBuffer message = new StringBuffer(e.getMessage());
609 
610 					Object sourceObject = ole.getSourceObject();
611 					if (sourceObject != null) {
612 						message.append(" (sourceObject is ");
613 						message.append(sourceObject.getClass().getName());
614 						message.append(")");
615 					}
616 
617 					e = new ServletException(message.toString(), rootCause);
618 				}
619 			}
620 
621 			logException(e);
622 			throw e;
623 		}
624 		return actionForward;
625 	}
626 
627 	private void logException(Exception e) {
628 		LOG.error("unhandled exception thrown by KualiRequestProcessor.processActionPerform", e);
629 	}
630 
631 	/**
632 	 * Checks for errors in the error map and transforms them to struts action
633 	 * messages then stores in the request.
634 	 */
635 	private void publishMessages(HttpServletRequest request) {
636 		MessageMap errorMap = GlobalVariables.getMessageMap();
637 		if (!errorMap.hasNoErrors()) {
638 			ErrorContainer errorContainer = new ErrorContainer(errorMap);
639 
640 			request.setAttribute("ErrorContainer", errorContainer);
641 			request.setAttribute(Globals.ERROR_KEY, errorContainer.getRequestErrors());
642 			request.setAttribute("ErrorPropertyList", errorContainer.getErrorPropertyList());
643 		}
644 		
645 		if (errorMap.hasWarnings()) {
646 			WarningContainer warningsContainer = new WarningContainer(errorMap);
647 			
648 			request.setAttribute("WarningContainer", warningsContainer);
649 			request.setAttribute("WarningActionMessages", warningsContainer.getRequestMessages());
650 			request.setAttribute("WarningPropertyList", warningsContainer.getMessagePropertyList());
651 		}
652 		
653 		if (errorMap.hasInfo()) {
654 			InfoContainer infoContainer = new InfoContainer(errorMap);
655 			
656 			request.setAttribute("InfoContainer", infoContainer);
657 			request.setAttribute("InfoActionMessages", infoContainer.getRequestMessages());
658 			request.setAttribute("InfoPropertyList", infoContainer.getMessagePropertyList());
659 		}
660 	}
661 
662 	/**
663 	 * Checks for messages in GlobalVariables and places list in request
664 	 * attribute.
665 	 */
666 	private void saveMessages(HttpServletRequest request) {
667 		if (!KNSGlobalVariables.getMessageList().isEmpty()) {
668 			request.setAttribute(KRADConstants.GLOBAL_MESSAGES, KNSGlobalVariables.getMessageList().toActionMessages());
669 		}
670 	}
671 
672 	/**
673 	 * Checks for messages in GlobalVariables and places list in request
674 	 * attribute.
675 	 */
676 	private void saveAuditErrors(HttpServletRequest request) {
677 		if (!KNSGlobalVariables.getAuditErrorMap().isEmpty()) {
678 			request.setAttribute(KNSConstants.AUDIT_ERRORS, KNSGlobalVariables.getAuditErrorMap());
679 		}
680 	}
681 
682 	/**
683 	 * A simple exception that allows us to wrap an exception that is thrown out
684 	 * of a transaction template.
685 	 */
686 	@SuppressWarnings("serial")
687 	private static class WrappedRuntimeException extends RuntimeException {
688 		public WrappedRuntimeException(Exception e) {
689 			super(e);
690 		}
691 	}
692 
693 	/**
694 	 * @return the sessionDocumentService
695 	 */
696 	public SessionDocumentService getSessionDocumentService() {
697 		if ( sessionDocumentService == null ) {
698 			sessionDocumentService = KNSServiceLocator.getSessionDocumentService();
699 		}
700 		return this.sessionDocumentService;
701 	}
702 
703 	/**
704 	 * @return the transactionManager
705 	 */
706 	public PlatformTransactionManager getTransactionManager() {
707 		if ( transactionManager == null ) {
708 			transactionManager = KRADServiceLocatorInternal.getTransactionManager();
709 		}
710 		return this.transactionManager;
711 	}
712 	
713 	private ActionForm createNewActionForm(ActionMapping mapping, HttpServletRequest request) {
714         String name = mapping.getName();
715         FormBeanConfig config = moduleConfig.findFormBeanConfig(name);
716         if (config == null) {
717             log.warn("No FormBeanConfig found under '" + name + "'");
718             return (null);
719         }
720         ActionForm instance = RequestUtils.createActionForm(config, servlet);
721         if ("request".equals(mapping.getScope())) {
722             request.setAttribute(mapping.getAttribute(), instance);
723         } else {
724             HttpSession session = request.getSession();
725             session.setAttribute(mapping.getAttribute(), instance);
726         }
727         return instance;
728 	}
729 }