1
2
3
4
5
6
7
8
9
10
11
12
13
14
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
52
53
54
55
56
57
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
102
103
104
105
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
141
142
143
144
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
156
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
168
169
170
171
172
173
174
175 protected String getUserDir() {
176 return System.getProperty("user.dir");
177 }
178
179
180
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
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
224
225
226
227
228
229
230
231
232
233
234
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
257
258 protected void startLifecycles(List<Lifecycle> lifecycles) throws Exception {
259 for (Lifecycle lifecycle : lifecycles) {
260 lifecycle.start();
261 }
262 }
263
264
265
266
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
288
289 protected List<Lifecycle> getSuiteLifecycles() {
290 List<Lifecycle> lifecycles = new LinkedList<Lifecycle>();
291
292
293
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
306
307 lifecycles.add(getTestHarnessSpringResourceLoader());
308
309
310
311
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
323
324 if (clearTables) {
325 lifecycles.add(new ClearDatabaseLifecycle());
326 }
327
328
329
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
348
349
350
351
352 protected Lifecycle getLoadApplicationLifecycle() {
353
354 return null;
355 }
356
357
358
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
375
376 protected void loadSuiteTestData() throws Exception {
377
378 }
379
380
381
382
383 protected void loadPerTestData() throws Exception {
384
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
407
408
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
422
423
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
442
443
444
445 protected abstract String getModuleName();
446
447 protected void setClearTables(boolean clearTables) {
448 this.clearTables = clearTables;
449 }
450
451 }