View Javadoc

1   /*
2    * Copyright 2005-2007 The Kuali Foundation
3    *
4    *
5    * Licensed under the Educational Community License, Version 2.0 (the "License");
6    * you may not use this file except in compliance with the License.
7    * You may obtain a copy of the License at
8    *
9    * http://www.opensource.org/licenses/ecl2.php
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  package org.kuali.rice.kew.engine;
18  
19  import static org.junit.Assert.assertEquals;
20  import static org.junit.Assert.assertNotNull;
21  import static org.junit.Assert.assertTrue;
22  
23  import java.io.Serializable;
24  
25  import javax.xml.namespace.QName;
26  
27  import org.junit.Test;
28  import org.kuali.rice.kew.api.WorkflowDocument;
29  import org.kuali.rice.kew.api.WorkflowDocumentFactory;
30  import org.kuali.rice.kew.api.WorkflowRuntimeException;
31  import org.kuali.rice.kew.engine.node.Branch;
32  import org.kuali.rice.kew.engine.node.BranchState;
33  import org.kuali.rice.kew.messaging.MessageServiceNames;
34  import org.kuali.rice.kew.messaging.exceptionhandling.DocumentMessageExceptionHandler;
35  import org.kuali.rice.kew.postprocessor.DefaultPostProcessor;
36  import org.kuali.rice.kew.postprocessor.DocumentRouteStatusChange;
37  import org.kuali.rice.kew.postprocessor.ProcessDocReport;
38  import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
39  import org.kuali.rice.kew.service.KEWServiceLocator;
40  import org.kuali.rice.kew.test.KEWTestCase;
41  import org.kuali.rice.kew.test.TestUtilities;
42  import org.kuali.rice.kew.util.KEWConstants;
43  import org.kuali.rice.ksb.api.KsbApiServiceLocator;
44  import org.kuali.rice.ksb.api.bus.support.JavaServiceDefinition;
45  import org.kuali.rice.ksb.messaging.service.KSBJavaService;
46  
47  public class StandardWorkflowEngineTest extends KEWTestCase {
48  
49  	protected void loadTestData() throws Exception {
50  		loadXmlFile("EngineConfig.xml");
51  	}
52  
53  	/**
54  	 * Tests that the proper state is set up on the root branch in the document
55  	 * to indicate that both PROCESSED and FINAL callbacks have been made into
56  	 * the post processor.
57  	 */
58  	@Test public void testSystemBranchState() throws Exception {
59  		// route the document to final
60  		WorkflowDocument document = WorkflowDocumentFactory.createDocument(getPrincipalIdForName("ewestfal"), "SimpleDocType");
61  		document.route("");
62  		assertTrue("Document should be final.", document.isFinal());
63  
64  		// now look at the branch state
65  		DocumentRouteHeaderValue routeHeader = KEWServiceLocator.getRouteHeaderService().getRouteHeader(document.getDocumentId());
66  		Branch rootBranch = routeHeader.getRootBranch();
67  		assertNotNull(rootBranch);
68  		BranchState processedBranchState = rootBranch.getBranchState(KEWConstants.POST_PROCESSOR_PROCESSED_KEY);
69  		BranchState finalBranchState = rootBranch.getBranchState(KEWConstants.POST_PROCESSOR_FINAL_KEY);
70  		assertNotNull(processedBranchState);
71  		assertNotNull(finalBranchState);
72  		assertEquals("true", processedBranchState.getValue());
73  		assertEquals("true", finalBranchState.getValue());
74  		assertEquals(1, TestPostProcessor.processedCount);
75  		assertEquals(1, TestPostProcessor.finalCount);
76  	}
77  
78  	/**
79  	 * Tests that a FINAL document can go into exception routing and recover
80  	 * properly while only calling the PROCESSED and FINAL callbacks once.
81  	 */
82  	@Test public void testFinalDocumentExceptionRoutingRecovery() throws Exception {
83  
84  		// route the document to final
85  		WorkflowDocument document = WorkflowDocumentFactory.createDocument(getPrincipalIdForName("ewestfal"), "SimpleDocType");
86  		document.route("");
87  		assertTrue("Document should be final.", document.isFinal());
88  		assertEquals(1, TestPostProcessor.processedCount);
89  		assertEquals(1, TestPostProcessor.finalCount);
90  
91  		// now queue up an exploder which should push the document into
92  		// exception routing
93  		JavaServiceDefinition serviceDef = new JavaServiceDefinition();
94  		serviceDef.setPriority(new Integer(1));
95  		serviceDef.setQueue(true);
96  		serviceDef.setRetryAttempts(0);
97  		serviceDef.setServiceInterface(KSBJavaService.class.getName());
98  		serviceDef.setServiceName(new QName("KEW", "exploader"));
99  		serviceDef.setService(new ImTheExploderProcessor());
100 
101 		serviceDef.setMessageExceptionHandler(DocumentMessageExceptionHandler.class.getName());
102 		serviceDef.validate();
103 		KsbApiServiceLocator.getServiceBus().publishService(serviceDef, true);
104 
105 		KSBJavaService exploderAsService = (KSBJavaService) MessageServiceNames.getServiceAsynchronously(new QName("KEW", "exploader"), KEWServiceLocator.getRouteHeaderService().getRouteHeader(document.getDocumentId()));
106 		exploderAsService.invoke("");
107 		// we need to make the exploder a service to get this going again...
108 		// SpringServiceLocator.getRouteQueueService().requeueDocument(document.getDocumentId(),
109 		// ImTheExploderProcessor.class.getName());
110 		// fail("Should have exploded!!!");
111 		TestUtilities.waitForExceptionRouting();
112 
113 		// the document should be in exception routing now
114 		document = WorkflowDocumentFactory.loadDocument(getPrincipalIdForName("ewestfal"), document.getDocumentId());
115 		assertTrue("Document should be in exception routing.", document.isException());
116 		assertEquals(1, TestPostProcessor.processedCount);
117 		assertEquals(1, TestPostProcessor.finalCount);
118 
119 		assertTrue("ewestfal should have a complete request.", document.isCompletionRequested());
120 		document.complete("");
121 
122 		// the document should be final once again
123 		document = WorkflowDocumentFactory.loadDocument(getPrincipalIdForName("ewestfal"), document.getDocumentId());
124 		assertTrue("Document should be final.", document.isFinal());
125 		assertEquals(1, TestPostProcessor.processedCount);
126 		assertEquals(1, TestPostProcessor.finalCount);
127 	}
128 
129 	public void tearDown() throws Exception {
130 	    try {
131 		TestPostProcessor.resetProcessedCount();
132 		TestPostProcessor.resetFinalCount();
133 	    } finally {
134 		super.tearDown();
135 	    }
136 	}
137 
138 	public static class TestPostProcessor extends DefaultPostProcessor {
139 
140 		public static int finalCount = 0;
141 
142 		public static int processedCount = 0;
143 
144 		public ProcessDocReport doRouteStatusChange(DocumentRouteStatusChange statusChangeEvent) throws Exception {
145 			if (KEWConstants.ROUTE_HEADER_PROCESSED_CD.equals(statusChangeEvent.getNewRouteStatus())) {
146 				processedCount++;
147 			} else if (KEWConstants.ROUTE_HEADER_FINAL_CD.equals(statusChangeEvent.getNewRouteStatus())) {
148 				finalCount++;
149 			}
150 			return new ProcessDocReport(true);
151 		}
152 
153 		public static void resetProcessedCount() {
154 			processedCount = 0;
155 		}
156 
157 		public static void resetFinalCount() {
158 			finalCount = 0;
159 		}
160 	}
161 
162 	public static class ImTheExploderProcessor implements KSBJavaService {
163 
164 		public void invoke(Serializable payLoad) {
165 			throw new WorkflowRuntimeException("I'm the Exploder!!!");
166 		}
167 
168 	}
169 
170 }