Подтвердить что ты не робот

PowerMockito (с Mockito) не работает с ExceptionInInitializerError

Мы используем Powermockito с Mockito, чтобы издеваться над некоторыми статическими классами. Кажется, что java.lang.ExceptionInInitializerError выбрасывается каждый раз.

Можете ли вы помочь мне определить, где проблема?

Проверяемый класс Java

package com.myproject.myproduct.search.domain;
import org.elasticsearch.index.query.MultiMatchQueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;

public class MyQueryBuilder {

    public MultiMatchQueryBuilder getMultiMatchQueryBuilder() {
        MultiMatchQueryBuilder builder = QueryBuilders.multiMatchQuery("term", "field1");
        builder.field("field1",200.9f);
        return builder;
    }
}

Испытание Junit с бегуном Powermock

package com.myproject.myproduct.search.domain;

import org.elasticsearch.index.query.MultiMatchQueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.powermock.api.mockito.PowerMockito;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;

@RunWith(PowerMockRunner.class)
@PrepareForTest(QueryBuilders.class)
public class MyQueryBuilderTest {

    private MyQueryBuilder myQueryBuilder;

    @Test
    public void test() {
        PowerMockito.mockStatic(QueryBuilders.class);
        MultiMatchQueryBuilder builder = PowerMockito.mock(MultiMatchQueryBuilder.class);
    }
}

Что это. Тест-код не работает, как только я пытаюсь высмеять MultiMatchQueryBuilder.

Это исключение:

java.lang.ExceptionInInitializerError at org.elasticsearch.common.logging.DeprecationLogger. (DeprecationLogger.java:138)   на org.elasticsearch.common.ParseField. (ParseField.java:35)   в org.elasticsearch.index.query.AbstractQueryBuilder. (AbstractQueryBuilder.java:53)   в sun.reflect.GeneratedSerializationConstructorAccessor7.newInstance(Неизвестно Источник) в java.lang.reflect.Constructor.newInstance(Constructor.java:423) в org.objenesis.instantiator.sun.SunReflectionFactoryInstantiator.newInstance(SunReflectionFactoryInstantiator.java:40)   на org.objenesis.ObjenesisBase.newInstance(ObjenesisBase.java:59) at org.mockito.internal.creation.jmock.ClassImposterizer.createProxy(ClassImposterizer.java:128)   в org.mockito.internal.creation.jmock.ClassImposterizer.imposterise(ClassImposterizer.java:63)   в org.powermock.api.mockito.internal.mockcreation.MockCreator.createMethodInvocationControl(MockCreator.java:111)   в org.powermock.api.mockito.internal.mockcreation.MockCreator.mock(MockCreator.java:60)   на org.powermock.api.mockito.PowerMockito.mock(PowerMockito.java:143)   в com.spartasystems.stratas.search.domain.MyQueryBuilderTest.testBoostSetProperly(MyQueryBuilderTest.java:22)   at sun.reflect.NativeMethodAccessorImpl.invoke0 (Нативный метод) в sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)   в sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)   в java.lang.reflect.Method.invoke(Method.java:498) в org.junit.internal.runners.TestMethod.invoke(TestMethod.java:68) в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl $PowerMockJUnit44MethodRunner.runTestMethod(PowerMockJUnit44RunnerDelegateImpl.java:310)   в org.junit.internal.runners.MethodRoadie $2.run(MethodRoadie.java:88)   в org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:96)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl $PowerMockJUnit44MethodRunner.executeTest(PowerMockJUnit44RunnerDelegateImpl.java:294)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit47RunnerDelegateImpl $PowerMockJUnit47MethodRunner.executeTestInSuper(PowerMockJUnit47RunnerDelegateImpl.java:127)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit47RunnerDelegateImpl $PowerMockJUnit47MethodRunner.executeTest(PowerMockJUnit47RunnerDelegateImpl.java:82)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl $PowerMockJUnit44MethodRunner.runBeforesThenTestThenAfters(PowerMockJUnit44RunnerDelegateImpl.java:282)   в org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.java:86)   на org.junit.internal.runners.MethodRoadie.run(MethodRoadie.java:49)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.invokeTestMethod(PowerMockJUnit44RunnerDelegateImpl.java:207)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.runMethods(PowerMockJUnit44RunnerDelegateImpl.java:146)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl $1.run(PowerMockJUnit44RunnerDelegateImpl.java:120)   в org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.java:33)   в org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:45)   в org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.run(PowerMockJUnit44RunnerDelegateImpl.java:122)   в org.powermock.modules.junit4.common.internal.impl.JUnit4TestSuiteChunkerImpl.run(JUnit4TestSuiteChunkerImpl.java:104)   в org.powermock.modules.junit4.common.internal.impl.AbstractCommonPowerMockRunner.run(AbstractCommonPowerMockRunner.java:53)   в org.powermock.modules.junit4.PowerMockRunner.run(PowerMockRunner.java:53)   на org.junit.runner.JUnitCore.run(JUnitCore.java:160) в com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)   в com.intellij.rt.execution.junit.IdeaTestRunner $Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)   в com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)   в com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70) Вызывается: java.lang.NullPointerException at org.elasticsearch.Build. (Build.java:47)... 41 подробнее

Процесс завершен с кодом выхода 255

Примечание:

Исходный код реальных классов elasticsearch можно найти здесь.

https://github.com/elastic/elasticsearch/blob/master/core/src/main/java/org/elasticsearch/index/query/QueryBuilders.java

и

https://github.com/elastic/elasticsearch/blob/master/core/src/main/java/org/elasticsearch/index/query/MultiMatchQueryBuilder.java

4b9b3361

Ответ 1

При вызове с помощью mocks org.elasticsearch.Build#getElasticsearchCodebase

Build.class.getProtectionDomain().getCodeSource().getLocation()

возвращает null, потому что код не имеет местоположения (динамический метод, сгенерированный cglib.)

Итак, когда инициализирует org.elasticsearch.Build во время вашего макета кода, используя

final URL url = getElasticsearchCodebase(); // url is null
final String urlStr = url.toString(); // null pointer exception.

Конечно, макет не будет успешным и бросит ExceptionInInitializerError, который указывает на исключение, возникшее при оценке статического инициализатора или инициализатора для статической переменной.


Вы можете легко воспроизвести это исключение, используя следующий код:

@RunWith(PowerMockRunner.class)
@PrepareForTest({QueryBuilders.class})
public class MyQueryBuilderTest {

    @Test
    public void test() {
        final Build current = Build.CURRENT;
    }

}