org.kuali.kfs.sys.context.KualiTestBase.java Source code

Java tutorial

Introduction

Here is the source code for org.kuali.kfs.sys.context.KualiTestBase.java

Source

/*
 * The Kuali Financial System, a comprehensive financial management system for higher education.
 * 
 * Copyright 2005-2014 The Kuali Foundation
 * 
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as
 * published by the Free Software Foundation, either version 3 of the
 * License, or (at your option) any later version.
 * 
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Affero General Public License for more details.
 * 
 * You should have received a copy of the GNU Affero General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
package org.kuali.kfs.sys.context;

import java.io.File;
import java.lang.reflect.Method;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

import javax.sql.DataSource;

import junit.framework.TestCase;

import org.apache.commons.lang.StringUtils;
import org.apache.log4j.Logger;
import org.apache.ojb.broker.OptimisticLockException;
import org.kuali.kfs.sys.ConfigureContext;
import org.kuali.kfs.sys.KualiTestConstants;
import org.kuali.kfs.sys.batch.service.CacheService;
import org.kuali.kfs.sys.batch.service.SchedulerService;
import org.kuali.kfs.sys.fixture.UserNameFixture;
import org.kuali.kfs.sys.service.ConfigurableDateService;
import org.kuali.rice.core.api.config.property.ConfigContext;
import org.kuali.rice.core.framework.persistence.jdbc.datasource.XAPoolDataSource;
import org.kuali.rice.krad.UserSession;
import org.kuali.rice.krad.exception.ValidationException;
import org.kuali.rice.krad.util.ErrorMessage;
import org.kuali.rice.krad.util.GlobalVariables;
import org.kuali.rice.ksb.util.KSBConstants;
import org.springframework.cache.annotation.CacheEvict;
import org.springframework.jdbc.CannotGetJdbcConnectionException;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.DefaultTransactionDefinition;
import org.springmodules.orm.ojb.OjbOperationException;

/**
 * This class should be extended by all Kuali unit tests.
 *
 * @see ConfigureContext
 * @see RelatesTo
 */

public abstract class KualiTestBase extends TestCase implements KualiTestConstants {
    private static final Logger LOG = Logger.getLogger(KualiTestBase.class);
    protected static boolean log4jConfigured = false;
    protected static RuntimeException configurationFailure;
    protected static boolean springContextInitialized = false;
    protected static boolean batchScheduleInitialized = false;
    protected static TransactionStatus transactionStatus;
    protected static UserNameFixture userSessionUsername;
    protected static UserSession userSession;
    protected static Set<String> generatedFiles = new HashSet<String>();

    /**
     * Determines whether to actually run the test using the RelatesTo annotation, onfigures the appropriate context using the
     * ConfigureContext annotation, and logs extra details if the test invocation's OJB operations happen to encounter an
     * OptimisticLockException or if this test has related Jiras.
     *
     * @throws Throwable
     */
    @Override
    public final void runBare() throws Throwable {

        final String testName = getClass().getName() + "." + getName();

        GlobalVariables.clear();
        ConfigureContext contextConfiguration = getMethod(getName()).getAnnotation(ConfigureContext.class) != null
                ? getMethod(getName()).getAnnotation(ConfigureContext.class)
                : getMethod("setUp").getAnnotation(ConfigureContext.class) != null
                        ? getMethod("setUp").getAnnotation(ConfigureContext.class)
                        : getClass().getAnnotation(ConfigureContext.class);
        if (contextConfiguration != null) {
            configure(contextConfiguration);
            SpringContext.getBean(ConfigurableDateService.class).setCurrentDate(new java.util.Date());
            ConfigContext.getCurrentContextConfig().putProperty(KSBConstants.Config.MESSAGE_DELIVERY,
                    KSBConstants.MESSAGING_SYNCHRONOUS);
        }

        if (!log4jConfigured) {
            Log4jConfigurer.configureLogging(false);
            log4jConfigured = true;
        }

        if (LOG.isInfoEnabled()) {
            LOG.info("Entering test '" + testName + "'");
        }
        try {
            setUp();
            try {
                runTest();
            } catch (OjbOperationException e) {
                // log more detail for OptimisticLockExceptions
                OjbOperationException ooe = e;
                Throwable cause = ooe.getCause();
                if (cause instanceof OptimisticLockException) {
                    OptimisticLockException ole = (OptimisticLockException) cause;
                    StringBuffer message = new StringBuffer("caught OptimisticLockException, caused by ");
                    Object sourceObject = ole.getSourceObject();
                    String suffix = null;
                    try {
                        // try to add instance details
                        suffix = sourceObject.toString();
                    } catch (Exception e2) {
                        // just use the class name
                        suffix = sourceObject.getClass().getName();
                    }
                    message.append(suffix);
                    LOG.error(message.toString());
                }
                throw e;
            } finally {
                tearDown();
                if (springContextInitialized) {
                    LOG.info("clearing caches");
                    clearAllCaches();
                }
            }
        } catch (ValidationException ex) {
            fail("Test threw an unexpected ValidationException: " + dumpMessageMapErrors());
        } catch (Throwable ex) {
            if (ex instanceof CannotGetJdbcConnectionException
                    || StringUtils.contains(ex.getMessage(), "GenericPool:checkOut")
                    || StringUtils.contains(ex.getMessage(), "no connection available")) {
                LOG.fatal("UNABLE TO OBTAIN DATABASE CONNECTION!  THIS AND MANY OTHER TESTS WILL LIKELY FAIL!", ex);
                DataSource ds = (DataSource) SpringContext.getBean("datasource");
                if (ds != null && ds instanceof XAPoolDataSource) {
                    LOG.fatal("Datasource Information:");
                    LOG.fatal(((XAPoolDataSource) ds).getDataSource().toString());
                }
                fail("CONFIGURATION ERROR: UNABLE TO OBTAIN DATABASE CONNECTION!");
            }
            throw ex;
        } finally {
            if (contextConfiguration != null) {
                endTestTransaction();
            }
            for (String filePath : generatedFiles) {
                try {
                    File file = new File(filePath);
                    if (file.exists()) {
                        file.delete();
                    }
                } catch (Exception e) {
                    LOG.warn("Unable to delete file: " + filePath, e);
                }
            }
            generatedFiles.clear();
            GlobalVariables.setUserSession(null);
            GlobalVariables.clear();
            if (LOG.isInfoEnabled()) {
                LOG.info("Leaving test '" + testName + "'");
            }
        }
    }

    @CacheEvict(allEntries = true, value = { "" })
    protected void clearAllCaches() {
        SpringContext.getBean(CacheService.class).clearSystemCaches();
    }

    protected void clearBoCache(Class boClass) {
        SpringContext.getBean(CacheService.class).clearKfsBusinessObjectCache(boClass);
    }

    protected void changeCurrentUser(UserNameFixture sessionUser) throws Exception {
        GlobalVariables.setUserSession(new UserSession(sessionUser.toString()));
    }

    protected void addGeneratedFile(String filePath) {
        generatedFiles.add(filePath);
    }

    /**
     *  Do not call this method!  It is used by the ContinuousIntegrationShutdown "test" to stop the context and clear its state.
     *  Any other use will likely break the CI environment.
     */
    protected void stopSpringContext() {
        if (springContextInitialized) {
            try {
                SpringContext.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
            springContextInitialized = false;
        }
    }

    private void configure(ConfigureContext contextConfiguration) throws Exception {
        if (configurationFailure != null) {
            throw configurationFailure;
        }
        if (!springContextInitialized) {
            try {
                KFSTestStartup.initializeKfsTestContext();
                springContextInitialized = true;
            } catch (RuntimeException e) {
                configurationFailure = e;
                throw e;
            }
        }
        if (!batchScheduleInitialized && contextConfiguration.initializeBatchSchedule()) {
            SpringContext.getBean(SchedulerService.class).initialize();
            batchScheduleInitialized = true;
        }
        if (!contextConfiguration.shouldCommitTransactions()) {
            LOG.info("Starting test transaction that will be rolled back");
            DefaultTransactionDefinition defaultTransactionDefinition = new DefaultTransactionDefinition();
            defaultTransactionDefinition.setTimeout(3600);
            //            defaultTransactionDefinition.setReadOnly(true);
            transactionStatus = getTransactionManager().getTransaction(defaultTransactionDefinition);
            transactionStatus.setRollbackOnly();
        } else {
            LOG.info("Test transaction not used");
            LOG.info("Starting transaction which will COMMIT at the end of the test");
            DefaultTransactionDefinition defaultTransactionDefinition = new DefaultTransactionDefinition();
            defaultTransactionDefinition.setTimeout(3600);
            transactionStatus = getTransactionManager().getTransaction(defaultTransactionDefinition);
            //            transactionStatus = null;
        }
        UserNameFixture sessionUser = contextConfiguration.session();
        if (sessionUser != UserNameFixture.NO_SESSION) {
            GlobalVariables.setUserSession(new UserSession(sessionUser.toString()));
        }
    }

    private void endTestTransaction() {
        if (transactionStatus != null) {
            if (transactionStatus.isRollbackOnly()) {
                LOG.info("rolling back transaction");
                try {
                    getTransactionManager().rollback(transactionStatus);
                } catch (Exception ex) {
                    LOG.warn("Error rolling back transaction", ex);
                }
            } else {
                LOG.info("committing test transaction");
                try {
                    getTransactionManager().commit(transactionStatus);
                } catch (Exception ex) {
                    LOG.warn("Error committing transaction", ex);
                }
            }
        }
    }

    protected PlatformTransactionManager getTransactionManager() {
        return (PlatformTransactionManager) SpringContext.getService("transactionManager");
    }

    private Method getMethod(String methodName) {
        Class<? extends Object> clazz = getClass();
        while (clazz != null) {
            try {
                return clazz.getDeclaredMethod(methodName);
            } catch (NoSuchMethodException e) {
                clazz = clazz.getSuperclass();
            }
        }
        throw new RuntimeException("KualiTestBase was unable to getMethod: " + methodName);
    }

    /**
     * This method is used during debugging to dump the contents of the error map, including the key names. It is not used by the
     * application in normal circumstances at all.
     */
    protected String dumpMessageMapErrors() {
        if (GlobalVariables.getMessageMap().hasNoErrors()) {
            return "";
        }

        StringBuilder message = new StringBuilder();
        for (String key : GlobalVariables.getMessageMap().getErrorMessages().keySet()) {
            List<ErrorMessage> errorList = GlobalVariables.getMessageMap().getErrorMessages().get(key);

            for (ErrorMessage em : errorList) {
                message.append(key).append(" = ").append(em.getErrorKey());
                if (em.getMessageParameters() != null) {
                    message.append(" : ");
                    String delim = "";
                    for (String parm : em.getMessageParameters()) {
                        message.append(delim).append("'").append(parm).append("'");
                        if ("".equals(delim)) {
                            delim = ", ";
                        }
                    }
                }
            }
            message.append('\n');
        }
        return message.toString();
    }
}