View Javadoc

1   /**
2    * Copyright 2005-2011 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.test;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.apache.log4j.Logger;
20  import org.apache.log4j.PropertyConfigurator;
21  import org.junit.After;
22  import org.junit.Before;
23  import org.kuali.rice.core.api.config.property.Config;
24  import org.kuali.rice.core.api.config.property.ConfigContext;
25  import org.kuali.rice.core.api.lifecycle.BaseLifecycle;
26  import org.kuali.rice.core.api.lifecycle.Lifecycle;
27  import org.kuali.rice.core.framework.resourceloader.SpringResourceLoader;
28  import org.kuali.rice.core.impl.config.property.JAXBConfigImpl;
29  import org.kuali.rice.test.data.PerSuiteUnitTestData;
30  import org.kuali.rice.test.lifecycles.PerSuiteDataLoaderLifecycle;
31  import org.springframework.core.io.FileSystemResourceLoader;
32  import org.springframework.core.io.Resource;
33  import org.springframework.core.io.ResourceLoader;
34  
35  import javax.xml.namespace.QName;
36  import java.io.File;
37  import java.io.IOException;
38  import java.util.ArrayList;
39  import java.util.Collections;
40  import java.util.HashSet;
41  import java.util.LinkedList;
42  import java.util.List;
43  import java.util.Properties;
44  import java.util.Set;
45  
46  import static org.junit.Assert.assertNotNull;
47  import static org.junit.Assert.fail;
48  
49  
50  /**
51   * Useful superclass for all Rice test cases. Handles setup of test utilities and a test environment. Configures the
52   * Spring test environment providing a template method for custom context files in test mode. Also provides a template method
53   * for running custom transactional setUp. Tear down handles automatic tear down of objects created inside the test
54   * environment.
55   * 
56   * @author Kuali Rice Team (rice.collab@kuali.org)
57   * @since 0.9
58   */
59  public abstract class RiceTestCase extends BaseRiceTestCase {
60  
61      private static final Logger LOG = Logger.getLogger(RiceTestCase.class);
62  
63      private static final String ALT_LOG4J_CONFIG_LOCATION_PROP = "alt.log4j.config.location";
64      private static final String DEFAULT_LOG4J_CONFIG = "classpath:rice-testharness-default-log4j.properties";
65      protected static final String DEFAULT_TEST_HARNESS_SPRING_BEANS = "classpath:TestHarnessSpringBeans.xml";
66      protected static boolean SUITE_LIFE_CYCLES_RAN = false;
67      protected static boolean SUITE_LIFE_CYCLES_FAILED = false;
68      protected static String failedSuiteTestName;
69  
70      protected List<Lifecycle> perTestLifeCycles = new LinkedList<Lifecycle>();
71  
72      protected List<Lifecycle> suiteLifeCycles = new LinkedList<Lifecycle>();
73  
74      private static Set<String> perSuiteDataLoaderLifecycleNamesRun = new HashSet<String>();
75  
76      private List<String> reports = new ArrayList<String>();
77  
78      private SpringResourceLoader testHarnessSpringResourceLoader;
79      private boolean clearTables = true;
80  
81      @Override
82  	@Before
83      public void setUp() throws Exception {
84          try {
85              configureLogging();
86              logBeforeRun();
87  
88              final long initTime = System.currentTimeMillis();
89  
90              setUpInternal();
91  
92              report("Time to start all Lifecycles: " + (System.currentTimeMillis() - initTime));
93          } catch (Throwable e) {
94              e.printStackTrace();
95              tearDown();
96              throw new RuntimeException(e);
97          }
98      }
99  
100     /**
101      * Internal setUp() implementation which is invoked by the main setUp() and wrapped with exception handling
102      *
103      * <p>Subclasses should override this method if they want to
104      * add set up steps that should occur in the standard set up process, wrapped by
105      * exception handling.</p>
106      */
107     protected void setUpInternal() throws Exception {
108         assertNotNull(getModuleName());
109         setModuleName(getModuleName());
110         setBaseDirSystemProperty(getModuleName());
111 
112         this.perTestLifeCycles = getPerTestLifecycles();
113         this.suiteLifeCycles = getSuiteLifecycles();
114 
115         if (SUITE_LIFE_CYCLES_FAILED) {
116         	fail("Suite Lifecycles startup failed on test " + failedSuiteTestName + "!!!  Please see logs for details.");
117         }
118         if (!SUITE_LIFE_CYCLES_RAN) {
119 	        try {
120     	        startLifecycles(this.suiteLifeCycles);
121         	    SUITE_LIFE_CYCLES_RAN = true;
122         	} catch (Throwable e) {
123         		e.printStackTrace();
124                 SUITE_LIFE_CYCLES_RAN = false;
125                 SUITE_LIFE_CYCLES_FAILED = true;
126                 failedSuiteTestName = getFullTestName();
127                 tearDown();
128                 stopLifecycles(this.suiteLifeCycles);
129                 throw new RuntimeException(e);
130             }
131         }
132 
133         startSuiteDataLoaderLifecycles();
134 
135         startLifecycles(this.perTestLifeCycles);
136 
137     }
138 
139     /**
140      * This block is walking up the class hierarchy of the current unit test looking for PerSuiteUnitTestData annotations. If it finds one,
141      * it will run it once, then add it to a set so that it does not get run again. This is needed so that multiple 
142      * tests can extend from the same suite and so that there can be multiple suites throughout the test source branch.
143      * 
144      * @throws Exception if a PerSuiteDataLoaderLifecycle is unable to be started
145      */
146     protected void startSuiteDataLoaderLifecycles() throws Exception {
147         List<Class> classes = TestUtilities.getHierarchyClassesToHandle(getClass(), new Class[] { PerSuiteUnitTestData.class }, perSuiteDataLoaderLifecycleNamesRun);
148         for (Class c: classes) {
149             new PerSuiteDataLoaderLifecycle(c).start();
150             perSuiteDataLoaderLifecycleNamesRun.add(c.getName());
151         }
152     }
153 
154     /**
155      * maven will set this property and find resources from the config based on it. This makes eclipse testing work because
156      * we have to put the basedir in our config files in order to find things when testing from maven
157      */
158     protected void setBaseDirSystemProperty(String moduleBaseDir) {
159         if (System.getProperty("basedir") == null) {
160         	final String userDir = System.getProperty("user.dir");
161         	
162             System.setProperty("basedir", userDir + ((userDir.endsWith(File.separator + "it" + File.separator + moduleBaseDir)) ? "" : File.separator + "it" + File.separator + moduleBaseDir));
163         }
164     }
165 
166     /**
167      * the absolute path on the file system to the root folder of the maven module containing a child of this class
168      * e.g. for krad: [rice-project-dir]/it/krad
169      *
170      * <p>
171      * the user.dir property can be set on the CLI or IDE run configuration e.g. -Duser.dir=/some/dir
172      * </p>
173      * @return the value of a system property 'user.dir' if it exists, null if not
174      */
175     protected String getUserDir() {
176         return System.getProperty("user.dir");
177     }
178 
179     /**
180      * Returns the basedir for the module under which the tests are currently executing.
181      */
182     protected String getBaseDir() {
183         return System.getProperty("basedir");
184     }
185 
186     protected void setModuleName(String moduleName) {
187         if (System.getProperty("module.name") == null) {
188             System.setProperty("module.name", moduleName);
189         }
190     }
191 
192     @Override
193 	@After
194     public void tearDown() throws Exception {
195     	// wait for outstanding threads to complete for 1 minute
196     	ThreadMonitor.tearDown(60000);
197         stopLifecycles(this.perTestLifeCycles);
198         logAfterRun();
199     }
200 
201     protected void logBeforeRun() {
202         LOG.info("##############################################################");
203         LOG.info("# Starting test " + getFullTestName() + "...");
204         LOG.info("# " + dumpMemory());
205         LOG.info("##############################################################");
206     }
207 
208     protected void logAfterRun() {
209         LOG.info("##############################################################");
210         LOG.info("# ...finished test " + getFullTestName());
211         LOG.info("# " + dumpMemory());
212         for (final String report : this.reports) {
213             LOG.info("# " + report);
214         }
215         LOG.info("##############################################################\n\n\n");
216     }
217     
218     protected String getFullTestName() {
219     	return getClass().getSimpleName() + "." + getName();
220     }
221 
222     /**
223      * configures logging using custom properties file if specified, or the default one.
224      * Log4j also uses any file called log4.properties in the classpath
225      *
226      * <p>To configure a custom logging file, set a JVM system property on using -D. For example
227      * -Dalt.log4j.config.location=file:/home/me/kuali/test/dev/log4j.properties
228      * </p>
229      *
230      * <p>The above option can also be set in the run configuration for the unit test in the IDE.
231      * To avoid log4j using files called log4j.properties that are defined in the classpath, add the following system property:
232      * -Dlog4j.defaultInitOverride=true
233      * </p>
234      * @throws IOException
235      */
236 	protected void configureLogging() throws IOException {
237         ResourceLoader resourceLoader = new FileSystemResourceLoader();
238         String altLog4jConfigLocation = System.getProperty(ALT_LOG4J_CONFIG_LOCATION_PROP);
239         Resource log4jConfigResource = null;
240         if (!StringUtils.isEmpty(altLog4jConfigLocation)) { 
241             log4jConfigResource = resourceLoader.getResource(altLog4jConfigLocation);
242         }
243         if (log4jConfigResource == null || !log4jConfigResource.exists()) {
244             System.out.println("Alternate Log4j config resource does not exist! " + altLog4jConfigLocation);
245             System.out.println("Using default log4j configuration: " + DEFAULT_LOG4J_CONFIG);
246             log4jConfigResource = resourceLoader.getResource(DEFAULT_LOG4J_CONFIG);
247         } else {
248             System.out.println("Using alternate log4j configuration at: " + altLog4jConfigLocation);
249         }
250         Properties p = new Properties();
251         p.load(log4jConfigResource.getInputStream());
252         PropertyConfigurator.configure(p);
253     }
254 
255 	/**
256 	 * Executes the start() method of each of the lifecycles in the given list.
257 	 */
258     protected void startLifecycles(List<Lifecycle> lifecycles) throws Exception {
259         for (Lifecycle lifecycle : lifecycles) {
260                 lifecycle.start();
261         }
262     }
263 
264     /**
265      * Executes the stop() method of each of the lifecyles in the given list.  The
266      * List of lifecycles is processed in reverse order.
267      */
268     protected void stopLifecycles(List<Lifecycle> lifecycles) throws Exception {
269         int lifecyclesSize = lifecycles.size() - 1;
270         for (int i = lifecyclesSize; i >= 0; i--) {
271             try {
272             	if (lifecycles.get(i) == null) {
273             		LOG.warn("Attempted to stop a null lifecycle");
274             	} else {
275             		if (lifecycles.get(i).isStarted()) {
276                         LOG.warn("Attempting to stop a lifecycle " + lifecycles.get(i).getClass());
277             			lifecycles.get(i).stop();
278             		}
279             	}
280             } catch (Exception e) {
281                 LOG.error("Failed to shutdown one of the lifecycles!", e);
282             }
283         }
284     }
285 
286     /**
287      * Returns the List of Lifecycles to start when the unit test suite is started
288      */
289     protected List<Lifecycle> getSuiteLifecycles() {
290         List<Lifecycle> lifecycles = new LinkedList<Lifecycle>();
291         
292         /**
293          * Initializes Rice configuration from the test harness configuration file.
294          */
295         lifecycles.add(new BaseLifecycle() {
296             @Override
297 			public void start() throws Exception {
298                 Config config = getTestHarnessConfig();
299                 ConfigContext.init(config);
300                 super.start();
301             }
302         });
303         
304         /**
305          * Loads the TestHarnessSpringBeans.xml file which obtains connections to the DB for us
306          */
307         lifecycles.add(getTestHarnessSpringResourceLoader());
308         
309         /**
310          * Establishes the TestHarnessServiceLocator so that it has a reference to the Spring context
311          * created from TestHarnessSpringBeans.xml
312          */
313         lifecycles.add(new BaseLifecycle() {
314             @Override
315 			public void start() throws Exception {
316                 TestHarnessServiceLocator.setContext(getTestHarnessSpringResourceLoader().getContext());
317                 super.start();
318             }
319         });
320         
321         /**
322          * Clears the tables in the database.
323          */
324         if (clearTables) {
325         	lifecycles.add(new ClearDatabaseLifecycle());
326         }
327         
328         /**
329          * Loads Suite Test Data
330          */
331         lifecycles.add(new BaseLifecycle() {
332         	@Override
333 			public void start() throws Exception {
334         		loadSuiteTestData();
335         		super.start();
336         	}
337         });
338         
339         Lifecycle loadApplicationLifecycle = getLoadApplicationLifecycle();
340         if (loadApplicationLifecycle != null) {
341         	lifecycles.add(loadApplicationLifecycle);
342         }
343         return lifecycles;
344     }
345     
346     /**
347      * This should return a Lifecycle that can be used to load the application
348      * being tested.  For example, this could start a Jetty Server which loads
349      * the application, or load a Spring context to establish a set of services,
350      * or any other application startup activities that the test depends upon.
351      */
352     protected Lifecycle getLoadApplicationLifecycle() {
353     	// by default return null, do nothing
354     	return null;
355     }
356 
357     /**
358      * @return Lifecycles run every test run
359      */
360     protected List<Lifecycle> getPerTestLifecycles() {
361     	List<Lifecycle> lifecycles = new LinkedList<Lifecycle>();
362         lifecycles.add(getPerTestDataLoaderLifecycle());
363         lifecycles.add(new BaseLifecycle() {
364             @Override
365 			public void start() throws Exception {
366                 loadPerTestData();
367                 super.start();
368             }
369         });
370         return lifecycles;
371     }
372     
373     /**
374      * A method that can be overridden to load test data for the unit test Suite.
375      */
376     protected void loadSuiteTestData() throws Exception {
377     	// do nothing by default, subclass can override
378     }
379     
380     /**
381      * A method that can be overridden to load test data on a test-by-test basis
382      */
383     protected void loadPerTestData() throws Exception {
384     	// do nothing by default, subclass can override
385     }
386 
387     protected void report(final String report) {
388         this.reports.add(report);
389     }
390 
391     protected String dumpMemory() {
392         final long total = Runtime.getRuntime().totalMemory();
393         final long free = Runtime.getRuntime().freeMemory();
394         final long max = Runtime.getRuntime().maxMemory();
395         return "[Memory] max: " + max + ", total: " + total + ", free: " + free;
396     }
397 
398     public SpringResourceLoader getTestHarnessSpringResourceLoader() {
399         if (testHarnessSpringResourceLoader == null) {
400             testHarnessSpringResourceLoader = new SpringResourceLoader(new QName("TestHarnessSpringContext"), getTestHarnessSpringBeansLocation(), null);
401         }
402         return testHarnessSpringResourceLoader;
403     }
404 
405     /**
406      * Returns the location of the test harness spring beans context file.
407      * Subclasses may override to specify a different location.
408      * @return the location of the test harness spring beans context file.
409      */
410     protected List<String> getTestHarnessSpringBeansLocation() {
411         return Collections.singletonList( DEFAULT_TEST_HARNESS_SPRING_BEANS );
412     }
413 
414     protected Config getTestHarnessConfig() throws Exception {
415         Config config = new JAXBConfigImpl(getConfigLocations(), System.getProperties());
416         config.parseConfig();
417         return config;
418     }
419 
420     /**
421      * Subclasses may override this method to customize the location(s) of the Rice configuration.
422      * By default it is: classpath:META-INF/" + getModuleName().toLowerCase() + "-test-config.xml"
423      * @return List of config locations to add to this tests config location.
424      */
425     protected List<String> getConfigLocations() {
426         List<String> configLocations = new ArrayList<String>();
427         configLocations.add(getRiceMasterDefaultConfigFile());
428         configLocations.add(getModuleTestConfigLocation());
429         return configLocations;
430     }
431     
432     protected String getModuleTestConfigLocation() {
433         return "classpath:META-INF/" + getModuleName().toLowerCase() + "-test-config.xml";
434     }
435 
436     protected String getRiceMasterDefaultConfigFile() {
437         return "classpath:META-INF/test-config-defaults.xml";
438     }
439 
440     /**
441      * same as the module directory in the project.
442      * 
443      * @return name of module that the tests located
444      */
445     protected abstract String getModuleName();
446 
447     protected void setClearTables(boolean clearTables) {
448     	this.clearTables = clearTables;
449     }
450     
451 }