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

การทดสอบครั้งแรกคือการทดสอบ getters และ setters ฉันกำลังเรียก constructor ที่นั่น (และมีข้อยกเว้นเกิดขึ้น) ส่วนที่สองคือการทดสอบคอนสตรัคเตอร์ ฉันต้องการควบคุมการแจงนับทรัพยากรที่มีในข้อมูลโค้ดที่สาม
วอย Reszelewski

1
ก่อนอื่นเรามาดูกันว่าการทดสอบของคุณนั้นแน่นมากกับการใช้งานของคุณ จากประสบการณ์สิ่งนี้จะนำไปสู่การทดสอบที่เปราะบาง โดยเฉพาะอย่างยิ่งคุณต้องการคิดว่า "กล่องดำ" เมื่อเขียนการทดสอบของคุณ "โค้ดนี้ควรทำอะไร" แทนที่จะเป็น "โค้ดนี้ทำหน้าที่อย่างไร" ประการที่สองฉันคิดว่าคุณน่าจะดีกว่าเพียงสร้างชุดของทรัพยากรและปล่อยให้ Java runtime จัดการกับ classloading ของตัวเอง
NilsH

มันเป็นไปได้ที่จะสร้างชุดของทรัพยากรต่าง ๆ เช่นกรณีการทดสอบ?
Wojciech Reszelewski

แน่ใจ วิธีที่ง่ายที่สุดสำหรับคุณน่าจะเป็นชื่อของทรัพยากร จากนั้นคุณสามารถส่งชื่อทรัพยากรที่แตกต่างกันในการทดสอบของคุณ
NilsH

คำตอบ:


408

ลองเพิ่มคำอธิบายประกอบนี้ในชั้นทดสอบของคุณ:

@PowerMockIgnore("javax.management.*")

ทำงานให้ฉัน


2
ความแม่นยำ * "เป็นคลาสทดสอบของคุณ" คำตอบที่ง่ายและมีประโยชน์!
pdem

3
สิ่งนี้สามารถทำได้ด้วยรหัสหรือ config เช่นกัน? ฉันไม่สามารถหาวิธีทำสิ่งนี้ได้ เรามีการทดสอบหลายร้อย ... ฉันไม่สามารถปรับได้ทั้งหมด
Frederic Leitenberger

1
@FredericLeitenberger ดูคำตอบของฉันด้านล่าง
user3474985

2
คุณช่วยอธิบายความตั้งใจและความหมายของการแก้ไขนี้ได้หรือไม่ เราให้คำแนะนำอะไรแก่ PowerMockito โดยใช้บรรทัดนั้น
Swapnil 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

2
อันนี้ช่วยฉัน: @PowerMockIgnore ({"javax.management. *", "org.apache.http. *", "com.amazonaws.http.conn.ssl. *", "javax.net.ssl. *" , "com.sun. *", "javax.xml. *", "javax.crypto. *"})
Fayaz Ahmed

26

ความขัดแย้งของClassloaderใช้สิ่งนี้:@PowerMockIgnore("javax.management.*")

ให้mock classloaderไม่โหลดjavax.*. มันใช้งานได้


หลังจากใช้@PowerMockIgnore("javax.management.*")งานคลาสการทดสอบจะทำงานได้ดีโดยลำพัง แต่การทำงานJunit testในแพ็คเกจนั้นมีFailed to load ApplicationContextข้อผิดพลาด org.apache.catalina.LifecycleException: A child container failed during startและอื่น ๆ
niaomingjian

8

นี่อาจเป็นหัวข้อเล็กน้อย แต่ฉันก็เจอปัญหานี้เช่นกัน ปรากฎว่าบางรุ่น Java ไม่สามารถจัดการ powermockito เมื่อ powermock พบว่ามี 2 คลาสที่มีชื่อเดียวกันในแพ็คเกจเดียวกัน (มากกว่าการพึ่งพาที่แตกต่างกัน)

สำหรับรุ่นที่สูงกว่า Java 7_25 ใด ๆ จะทำให้เกิดข้อผิดพลาดนี้


2
"ด้วยรุ่นใด ๆ ที่สูงกว่า Java 7_25 จะให้ข้อผิดพลาดนี้" นี่เป็นข้อมูล
Kajal Sinha

หมายความว่าอะไร: "ไม่สามารถจัดการ powermockito" ได้? มีวิธีใดที่จะจัดการกับมันได้โดยไม่สนใจคำอธิบายประกอบ?
บรรทัดที่

นานมาแล้ว แต่ฉันคิดว่าเราแยกมันออกโดยทำให้แน่ใจว่าไม่มี 2 คลาสที่มีชื่อเดียวกันในแพ็คเกจประเภทเดียวกัน แน่นอนถ้าคุณมีห้องสมุด 2 แห่งคุณต้องพึ่งพาและพวกมันอยู่ในนั้น ... มันจะเป็นเรื่องยาก ฉันไม่ทราบว่าปัญหานี้ได้รับการแก้ไขในระหว่างนี้หรือไม่
Rens Groenveld

3

Thread.classเพื่อที่จะเยาะเย้ยเรียนระบบเตรียมความพร้อมระดับที่เป็นเป้าหมายของการทดสอบที่ไม่ ไม่มีทางที่ PowerMock จะสามารถเครื่องดนตรีได้Thread.classเนื่องจากมันเป็นสิ่งจำเป็นในระหว่างการเริ่มต้น JVM - ก่อนที่ PowerMock สามารถใช้เครื่องดนตรีได้

วิธีการทำงานของเครื่องมือวัดเมื่อโหลดคลาสแล้วจะไม่สามารถทำการติดตั้งได้อีกต่อไป

ดูวิกิพีเดีย PowerMock


3

ใน PowerMock 1.7.0 การกำหนดค่าส่วนกลางที่ผู้ใช้กำหนดสามารถเพิ่มลงใน classpath ของโครงการของคุณ PowerMockConfig

org/powermock/extensions/configuration.properties

เพียงเพิ่มบรรทัดในไฟล์คุณสมบัติเช่น:

powermock.global-ignore=javax.management.*

สิ่งนี้จะแก้ไขข้อผิดพลาดสำหรับคลาสการทดสอบทั้งหมดในโครงการของคุณ

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.