All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.powermock.api.mockito.mockpolicies.Slf4jMockPolicy Maven / Gradle / Ivy

/*
 *   Copyright 2016 the original author or authors.
 *
 *   Licensed under the Apache License, Version 2.0 (the "License");
 *   you may not use this file except in compliance with the License.
 *   You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 *   Unless required by applicable law or agreed to in writing, software
 *   distributed under the License is distributed on an "AS IS" BASIS,
 *   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *   See the License for the specific language governing permissions and
 *   limitations under the License.
 *
 */

package org.powermock.api.mockito.mockpolicies;

import org.mockito.Mockito;
import org.powermock.core.spi.PowerMockPolicy;
import org.powermock.mockpolicies.MockPolicyClassLoadingSettings;
import org.powermock.mockpolicies.MockPolicyInterceptionSettings;
import org.powermock.mockpolicies.support.LogPolicySupport;

import java.lang.reflect.Method;

/**
 * Sfl4j mock policy that injects a Mockito-created mock to be returned on calls to getLogger factory methods.
 * The implementation returns a single mock instance per thread but it doesn't return a different mock instance based
 * on the actual value passed to getLogger. This limitation is acceptable in most real uses cases.
 * 

* Tests that want to do verifications on the mocked logger can do so by getting the mocked instance as production code * does: {@code org.slf4j.LoggerFactory.getLogger(Class)}. However, it is critical that the mocked logger is * reset after each test in order to avoid crosstalk between test cases. *

* * @author Alexandre Normand */ public class Slf4jMockPolicy implements PowerMockPolicy { private static final String LOGGER_FACTORY_CLASS_NAME = "org.slf4j.LoggerFactory"; private static final String LOGGER_FACTORY_METHOD_NAME = "getLogger"; private static final String FRAMEWORK_NAME = "sfl4j"; private static final String LOGGER_CLASS_NAME = "org.slf4j.Logger"; private static ThreadLocal threadLogger = new ThreadLocal(); @Override public void applyClassLoadingPolicy(MockPolicyClassLoadingSettings mockPolicyClassLoadingSettings) { mockPolicyClassLoadingSettings.addFullyQualifiedNamesOfClassesToLoadByMockClassloader( LOGGER_FACTORY_CLASS_NAME, "org.apache.log4j.Appender", "org.apache.log4j.xml.DOMConfigurator"); } @Override public void applyInterceptionPolicy(MockPolicyInterceptionSettings mockPolicyInterceptionSettings) { LogPolicySupport logPolicySupport = new LogPolicySupport(); Method[] loggerFactoryMethods = logPolicySupport.getLoggerMethods(LOGGER_FACTORY_CLASS_NAME, LOGGER_FACTORY_METHOD_NAME, FRAMEWORK_NAME); initializeMockForThread(logPolicySupport); for (Method loggerFactoryMethod : loggerFactoryMethods) { mockPolicyInterceptionSettings.stubMethod(loggerFactoryMethod, threadLogger.get()); } } private void initializeMockForThread(LogPolicySupport logPolicySupport) { Class loggerClass = getLoggerClass(logPolicySupport); if (threadLogger.get() == null) { /* * When mocking with Mockito we need to change the context CL to the same CL that is loading Mockito * otherwise the Mockito plugin mechanism will load the PowerMockMaker from the wrong classloader. */ final ClassLoader originalCl = Thread.currentThread().getContextClassLoader(); final ClassLoader classLoader = Mockito.class.getClassLoader(); Thread.currentThread().setContextClassLoader(classLoader); final Object mock; try { mock = Mockito.mock(loggerClass); } finally { Thread.currentThread().setContextClassLoader(originalCl); } threadLogger.set(mock); } } private Class getLoggerClass(LogPolicySupport logPolicySupport) { Class loggerType; try { loggerType = logPolicySupport.getType(LOGGER_CLASS_NAME, FRAMEWORK_NAME); } catch (RuntimeException e) { throw e; } catch (Exception e) { throw new RuntimeException(e); } return loggerType; } }