Java NullPointerException с Mocked объектом - PullRequest
0 голосов
/ 10 октября 2018

У меня есть модульный тест, который я пишу для класса, и этот класс использует ConfigurationManager, который я написал, чтобы получить конфигурацию из объекта TOML (и, следовательно, из файла TOML).

Код

Вот соответствующий код ConfigurationManager:

public class ConfigurationManager {
    // DEFAULT_LOGGING_LEVEL should represent level to use in production environment.
    private final static Level DEFAULT_LOGGING_LEVEL = Level.FINE;
    private final static String LOG_LEVEL_ENV_PROPERTY = "LOG_LEVEL";
    private final static String TOML_FILE_NAME = "config.toml";
    private final static String LOCAL_ENV_PROPERTY = "local";
    private final static String ENV_PROPERTY = "MY_ENV";
    private static Logger CM_LOGGER;

    private Environment environment;

    public ConfigurationManager() {
        environment = new Environment();
        CM_LOGGER = new LoggerUtil(ConfigurationManager.class.getName(), getLoggingLevel()).getLogger();
    }

    public File getTomlFile() throws URISyntaxException, FileNotFoundException {
        URI uri = getConfigURI();
        File tomlFile = new File(uri);
        if (!tomlFile.exists()) {
            String err = TOML_FILE_NAME + " does not exist!";
            CM_LOGGER.severe(err);
            throw new FileNotFoundException(err);
        }
        return tomlFile;
    }

    /**
     * @return A URI representing the path to the config
     * @throws URISyntaxException if getConfigURI encounters bad URI syntax.
     */
    private URI getConfigURI() throws URISyntaxException {
        return getClass().getClassLoader().getResource(TOML_FILE_NAME).toURI();
    }

    /**
     * Method for getting the app configuration as a toml object.
     *
     * @return A toml object built from the config.toml file.
     * @throws URISyntaxException    if getConfigURI encounters bad URI syntax.
     * @throws FileNotFoundException if getTomlFile can't find the config.toml file.
     */
    public Toml getConfigToml() throws URISyntaxException, FileNotFoundException {
        return new Toml().read(getTomlFile());
    }
}

А вот код, который вызывает данный Configuration Manager:

public class Listener {
    // Initializations
    private static final String CONFIG_LISTENER_THREADS = "listenerThreads";
    private static final String DEFAULT_LISTENER_THREADS = "1";

    /**
     * Constructor for getting app properties and initializing executor service with thread pool based on app prop.
     */
    @PostConstruct
    void init() throws FileNotFoundException, URISyntaxException {
        ConfigurationManager configurationManager = new ConfigurationManager();
            int listenerThreads = Integer.parseInt(configurationManager.getConfigToml()
                    .getString(CONFIG_LISTENER_THREADS, DEFAULT_LISTENER_THREADS));  
        this.executorService = Executors.newFixedThreadPool(listenerThreads);
        LOGGER.config("Listener executorService threads: " + listenerThreads);
    }
...
}

А вот тест для этого кода (см. Комментарий, чтобы увидеть, где срабатывает NPE):

public class ListenerTests {

    @Mock
    ConfigurationManager mockCM;

    @InjectMocks
    Listener listener = new Listener();

    @Before
    public void setUp(){
        MockitoAnnotations.initMocks(this);
    }

    @Test
    public void testListener_ShouldInit() throws FileNotFoundException, URISyntaxException {
        when(mockCM.getConfigToml().getString(any(), any())).thenReturn("5"); // !!!NPE TRIGGERED BY THIS LINE!!!
        listener.init();
        verify(mockCM, times(1)).getConfigToml().getString(any(), any());
    }
}

Проблема

Я получаюNullPointerException следующим образом

java.lang.NullPointerException
    at com.bose.source_account_listener.ListenerTests.testListener_ShouldInit(ListenerTests.java:37)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
    at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)

У меня есть предположение относительно проблемы здесь, но я не уверен, как проверить мою догадку.Я предполагаю, что я издеваюсь над ConfigurationManager, но макет не знает, что ConfigurationManager может сгенерировать файл TOML с методом getString, поэтому я в итоге получаю NPE.Это заставляет меня думать, что мне нужен фиктивный TOML для моего фиктивного ConfigurationManager, но я не уверен ни в этом, ни в том, что это правильное решение.

Я новичок в Mockito и Java в целом, поэтому любая помощь будет оценена.

Ответы [ 2 ]

0 голосов
/ 10 октября 2018

Вы пытаетесь смоделировать метод того, что возвращается, вызывая getConfigToml.То есть вам нужно смоделировать объект того, что должно быть возвращено getConfigToml, а затем вызвать getString для этого объекта.

0 голосов
/ 10 октября 2018

Вам нужно высмеивать каждый бит вашего звонка, а не только полный звонок.mockCM.getConfigToml() не имеет ложного ответа, поэтому возвращает ноль, и toString вызывается для нулевого объекта.Вы можете вернуть немодальный «Toml» или смоделировать Toml, а затем установить его с конфигурацией when.

Toml tomlMock = Mockito.mock(Toml.class);
when(mockCM.getConfigToml()).thenReturn(tmolMock);
when(tmolMock.getString(any(), any())).thenReturn("5"); 
...