web-dev-qa-db-ja.com

システムクラスのモック中にMockito + PowerMock LinkageError

私はそのようなコードスニペットを持っています:

@RunWith(PowerMockRunner.class)
@PrepareForTest({Thread.class})
public class AllMeasuresDataTest {

@Before
public void setUp() throws Exception {
}

@Test
public void testGetMeasures() {
    AllMeasuresData measure = new AllMeasuresData();
    assertEquals(measure.getMeasures(), null);
    HashMap<String, Measure> map = new HashMap<String, Measure>();
    measure.setMeasures(map);
    assertEquals(measure.getMeasures(), map);
    measure.setMeasures(null);
    assertEquals(measure.getMeasures(), null);
}

@Test
public void testAllMeasuresData() throws IOException {
    ClassLoader loader = PowerMockito.mock(ClassLoader.class);
    Thread threadMock = PowerMockito.mock(Thread.class);
    Vector<URL> vec = new Vector<URL>();
    Mockito.when(loader.getResources("measure")).thenReturn(vec.elements());
    Mockito.when(threadMock.getContextClassLoader()).thenReturn(loader);
    PowerMockito.mockStatic(Thread.class);
    Mockito.when(Thread.currentThread()).thenReturn(threadMock);
        ...
    }
}

このテストの実行中に私は得ました:

Java.lang.LinkageError: loader constraint violation: loader (instance of org/powermock/core/classloader/MockClassLoader) previously initiated loading for a different type with name "javax/management/MBeanServer"
at Java.lang.ClassLoader.defineClass1(Native Method)
at Java.lang.ClassLoader.defineClass(ClassLoader.Java:791)
at Java.lang.ClassLoader.defineClass(ClassLoader.Java:634)
at org.powermock.core.classloader.MockClassLoader.loadUnmockedClass(MockClassLoader.Java:201)
at org.powermock.core.classloader.MockClassLoader.loadModifiedClass(MockClassLoader.Java:149)
at org.powermock.core.classloader.DeferSupportingClassLoader.loadClass(DeferSupportingClassLoader.Java:67)
at Java.lang.ClassLoader.loadClass(ClassLoader.Java:356)
at org.codecover.instrumentation.Java.measurement.ProtocolImpl.initializeMBean(ProtocolImpl.Java:247)
at org.codecover.instrumentation.Java.measurement.ProtocolImpl.<init>(ProtocolImpl.Java:237)
at org.codecover.instrumentation.Java.measurement.ProtocolImpl.getInstance(ProtocolImpl.Java:185)
at measure.CodeCoverCoverageCounter$6ya5ud0ow79ijrr1dvjrp4nxx60qhxeua02ta2fzpmb1d.<clinit>(MeasureCalculatorsHolder.Java:146)
at measure.MeasureCalculatorsHolder.<clinit>(MeasureCalculatorsHolder.Java:17)
at Java.lang.Class.forName0(Native Method)
at Java.lang.Class.forName(Class.Java:188)
at javassist.runtime.Desc.getClassObject(Desc.Java:43)
at javassist.runtime.Desc.getClassType(Desc.Java:152)
at javassist.runtime.Desc.getType(Desc.Java:122)
at javassist.runtime.Desc.getType(Desc.Java:78)
at algorithm.AllMeasuresDataTest.testGetMeasures(AllMeasuresDataTest.Java:26)
at Sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at Sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.Java:57)
at Sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.Java:43)
at Java.lang.reflect.Method.invoke(Method.Java:601)
at org.junit.internal.runners.TestMethod.invoke(TestMethod.Java:66)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.runTestMethod(PowerMockJUnit44RunnerDelegateImpl.Java:312)
at org.junit.internal.runners.MethodRoadie$2.run(MethodRoadie.Java:86)
at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.Java:94)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.executeTest(PowerMockJUnit44RunnerDelegateImpl.Java:296)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit49RunnerDelegateImpl$PowerMockJUnit49MethodRunner.executeTestInSuper(PowerMockJUnit49RunnerDelegateImpl.Java:116)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit49RunnerDelegateImpl$PowerMockJUnit49MethodRunner.executeTest(PowerMockJUnit49RunnerDelegateImpl.Java:77)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.runBeforesThenTestThenAfters(PowerMockJUnit44RunnerDelegateImpl.Java:284)
at org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.Java:84)
at org.junit.internal.runners.MethodRoadie.run(MethodRoadie.Java:49)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.invokeTestMethod(PowerMockJUnit44RunnerDelegateImpl.Java:209)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.runMethods(PowerMockJUnit44RunnerDelegateImpl.Java:148)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$1.run(PowerMockJUnit44RunnerDelegateImpl.Java:122)
at org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.Java:34)
at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.Java:44)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.run(PowerMockJUnit44RunnerDelegateImpl.Java:120)
at org.powermock.modules.junit4.common.internal.impl.JUnit4TestSuiteChunkerImpl.run(JUnit4TestSuiteChunkerImpl.Java:101)
at org.powermock.modules.junit4.common.internal.impl.AbstractCommonPowerMockRunner.run(AbstractCommonPowerMockRunner.Java:53)
at org.powermock.modules.junit4.PowerMockRunner.run(PowerMockRunner.Java:53)
at org.Eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.Java:50)
at org.Eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.Java:38)
at org.Eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.Java:467)
at org.Eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.Java:683)
at org.Eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.Java:390)
at org.codecover.juniteclipse.runner.EclipseTestRunner.main(EclipseTestRunner.Java:40)

これを防ぐ方法を知っていますか?このようなコードをモックする別の方法があるかもしれません。

ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
...
Enumeration<URL> resources = classLoader.getResources(path);
141

この注釈をテストクラスに追加してみてください。

@PowerMockIgnore("javax.management.*")

私のために働いた。

357
crandrad

クラスローダー競合、これを使用:@PowerMockIgnore("javax.management.*")

模擬クラスローダーjavax.*.をロードしないようにします。

26
烬哥哥

ここで受け入れられた応答と同様に、私はすべてのSSL関連クラスを除外しなければならなくなりました。

@PowerMockIgnore({"javax.management.*", "org.Apache.http.conn.ssl.*", "com.amazonaws.http.conn.ssl.*", "javax.net.ssl.*"})

それをクラスの先頭に追加すると、エラーが解決しました。

23
Jason D

これは少し古いトピックかもしれませんが、私もこの問題に遭遇しました。 powermockが同じパッケージ内に同じ名前の2つのクラスがある(異なる依存関係にある)ことを検出した場合、いくつかのJavaバージョンはpowermockitoを処理できません。

Java 7_25以降のバージョンでは、このエラーが発生します。

7
Rens Groenveld

システムクラスを模擬するには、Thread.classではなく、テストの対象となるクラスを準備します。 PowerMockがThread.classをインスツルメントできる方法はありません。これは、JVMの起動中に必要になるためです-PowerMockがインスツルメントできる前に。

インストルメンテーションの動作方法。クラスがロードされると、インストルメント化できなくなります。

PowerMock wiki を参照してください。

3
ash

PowerMock 1.7.0では、ユーザー定義のグローバル構成をプロジェクトのクラスパスに追加できます。 PowerMockConfig

org/powermock/extensions/configuration.properties

プロパティファイルに次のような行を追加するだけです。

powermock.global-ignore=javax.management.*

これにより、プロジェクト内のすべてのテストクラスのエラーが解決されます。

3
user3474985