시스템 클래스를 조롱하는 동안 Mockito + PowerMock LinkageError


166

나는 그런 코드 스 니펫을 가지고있다 :

@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);

당신이 조롱하려는 것은 무엇입니까? 그리고 왜?
NilsH

첫 번째 테스트는 getter 및 setter 테스트이며 생성자를 호출합니다 (예외가 발생합니다). 두 번째는 생성자 테스트입니다. 세 번째 코드 스 니펫에 포함 된 리소스 열거를 제어하고 싶습니다.
Wojciech Reszelewski

1
우선, 테스트가 구현과 매우 밀접하게 결합되어있는 것으로 보입니다. 경험상, 이것은 깨지기 쉬운 테스트로 이어질 것입니다. 테스트를 작성할 때 "블랙 박스"를 생각하는 것이 좋습니다. "이 코드는 어떻게해야합니까"가 아니라 "이 코드는 어떻게해야합니까?" 둘째, 리소스 세트를 만들고 Java 런타임이 클래스 로딩 자체를 처리하게하는 것이 좋습니다.
NilsH

테스트 사례에 따라 다양한 리소스 세트를 생성 할 수 있습니까?
Wojciech Reszelewski

확실한. 가장 쉬운 방법은 아마도 리소스 이름을 매개 변수화하는 것입니다. 그런 다음 다른 리소스 이름을 테스트에 전달할 수 있습니다.
NilsH

답변:


408

이 주석을 테스트 클래스에 추가하십시오.

@PowerMockIgnore("javax.management.*")

나를 위해 일했다.


2
정밀도 * "테스트 클래스". 간단하고 유용한 답변!
pdem November

3
코드 또는 구성으로도 수행 할 수 있습니까? 나는 이것을 할 수있는 방법을 찾지 못했습니다. 우리는 수백 개의 테스트를 가지고 있습니다 ... 나는 그것들을 모두 조정할 수는 없습니다.
Frederic Leitenberger

1
@FredericLeitenberger 아래 내 답변을 참조하십시오
user3474985

2
이 수정의 직관과 의미를 설명해 주시겠습니까? 우리는 그 라인을 사용하여 PowerMockito에게 어떤 지시를하고 있습니까?
스왑 닐 B.

33

여기에 허용 된 응답과 비슷하게 SSL 관련 클래스를 모두 제외해야했습니다.

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

클래스 상단에 추가하면 오류가 해결되었습니다.


5
아직도 길을 더 추가해야했지만 내 생명을 구했습니다! @PowerMockIgnore({"javax.management.*", "org.apache.http.conn.ssl.*", "com.amazonaws.*", "javax.net.ssl.*","com.sun.*"})
Francisco López-Sancho

com.sun에 대해서도 알아두면 좋습니다.
Jason D

1
@PowerMockIgnore ({ "javax.management. *", "javax.crypto. *"})
Kristof Neirynck 's

2
이것은 저를 구했습니다 : @PowerMockIgnore ({ "javax.management. *", "org.apache.http. *", "com.amazonaws.http.conn.ssl. *", "javax.net.ssl. *" , "com.sun. *", "javax.xml. *", "javax.crypto. *"})
Fayaz Ahmed

26

클래스 로더 충돌, 이것을 사용하십시오 :@PowerMockIgnore("javax.management.*")

하자 모의 클래스 로더가 로드되지 않는 javax.*. 의미가 있습니다.


를 사용한 후 @PowerMockIgnore("javax.management.*")테스트 클래스는 단독으로 잘 작동합니다. 그러나 Junit test해당 패키지 에서 실행하면 Failed to load ApplicationContext오류 가 발생했습니다. org.apache.catalina.LifecycleException: A child container failed during start등등.
niaomingjian 10

8

이것은 오래된 주제 일 수도 있지만이 문제에 부딪 쳤습니다. powermock이 동일한 패키지에 동일한 이름을 가진 2 개의 클래스가 있다는 것을 알게되면 일부 Java 버전에서 powermockito를 처리 할 수 ​​없습니다 (다른 종속성을 통해).

Java 7_25보다 높은 버전에서는이 오류가 발생합니다.


2
"Java 7_25보다 높은 버전에서는이 오류가 발생합니다.", 이것은 유익합니다.
Kajal Sinha

무슨 의미 : "powermockito를 처리 할 수 ​​없습니다"? 주석으로 무시하는 것 외에 그것을 처리 할 수있는 방법이 있습니까?
라인

오래 전 일이지만 같은 종류의 패키지에 같은 이름의 클래스가 두 개가 없도록 정렬했습니다. 물론, 두 개의 라이브러리가 있고 거기에 상주하는 경우 어려울 것입니다. 그동안이 문제가 해결되었는지 모르겠습니다.
Rens Groenveld의

3

시스템 클래스를 조롱하려면 테스트 대상이 아닌 클래스를 준비하십시오 Thread.class. PowerMock이 계측을 Thread.class시작하기 훨씬 전에 JVM 시작 중에 필요하므로 PowerMock이 계측 할 수있는 방법은 없습니다 .

계측이 작동하는 방식은 일단 클래스가로드되면 더 이상 설치 될 수 없습니다.

PowerMock 위키를 참조하십시오 .


3

PowerMock 1.7.0에서는 사용자 정의 전역 구성을 프로젝트의 클래스 경로에 추가 할 수 있습니다. PowerMockConfig

org/powermock/extensions/configuration.properties

속성 파일에 다음과 같은 행을 추가하면됩니다.

powermock.global-ignore=javax.management.*

프로젝트의 모든 테스트 클래스에 대한 오류가 해결됩니다.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.