กระบวนการขัดข้องระหว่างการสร้างหัวฉีด RoboGuice หากมีอินสแตนซ์จำลองในโมดูลใด ๆ


85

ฉันมีปัญหากับการใช้เฟรมเวิร์ก RoboGuice และ AndroidMock ในการทดสอบหน่วย ฉันได้สร้างโครงการง่ายๆเพื่อแสดงปัญหาของฉัน ที่นี่ฉันสร้างอินสแตนซ์จำลองและลงทะเบียนใน RoboGuice แต่กระบวนการขัดข้องระหว่างเมธอด "setUp ()" และ "test01 ()" ตามที่ฉันเดาจริงๆแล้วกระบวนการหยุดทำงานเมื่อสร้าง Injector หากโมดูลใดมีอินสแตนซ์จำลองอยู่ภายใน

ถ้าฉันแทนที่อินสแตนซ์ที่ถูกเยาะเย้ยด้วยอินสแตนซ์ของคลาสที่ใช้อินเทอร์เฟซทุกอย่างก็จะทำงานได้ดี

มีใครรู้วิธีแก้ไขปัญหานี้หรือไม่?

นี่คือรหัสทดสอบของฉัน:

public class testInjectMock extends RoboUnitTestCase<MyApplication> {
    protected void setUp() throws Exception {
        InterfaceToMock instance = AndroidMock.createNiceMock(InterfaceToMock.class);           AndroidMock.expect(instance.SimpleMethod()).andStubReturn("Hello!");            
        MyModule myMockModule = new MyModule();
        myMockModule.setMockedInstance(instance);//Comment this string to get into the test01() method          
        MyApplication.setMyModule(myMockModule);
        super.setUp();
    }
    public void test01() {
        //It never comes here
    }
}

ซอร์สโค้ดโมดูล:

public class MyModule extends AbstractAndroidModule {
        protected InterfaceToMock mockedInstance;
        public void setMockedInstance(InterfaceToMock mockedInstance) {
            this.mockedInstance = mockedInstance;
        }
        @Override
        protected void configure() {
            if(mockedInstance != null)
                bind(InterfaceToMock.class).toInstance(mockedInstance);
        }
    }

เอาต์พุต logcat:

05-23 16:17:07.135: INFO/DEBUG(27): Build fingerprint: 'generic/sdk/generic/:2.1-update1/ECLAIR/35983:eng/test-keys'
05-23 16:17:07.135: INFO/DEBUG(27): pid: 2025, tid: 2031  >>> InjectMock.test <<<
05-23 16:17:07.145: INFO/DEBUG(27): signal 11 (SIGSEGV), fault addr 00000000
05-23 16:17:07.155: INFO/DEBUG(27):  r0 0011b218  r1 43d1caa0  r2 00000000  r3 00000000
05-23 16:17:07.155: INFO/DEBUG(27):  r4 43d1caa0  r5 0011b218  r6 451c0e30  r7 4000a958
05-23 16:17:07.155: INFO/DEBUG(27):  r8 ad00f380  r9 00138de0  10 426bda34  fp 00138de0
05-23 16:17:07.155: INFO/DEBUG(27):  ip 00000002  sp 451c0dc0  lr ad05ad1d  pc ad05a804  cpsr 00000030
05-23 16:17:07.295: INFO/DEBUG(27):          #00  pc 0005a804  /system/lib/libdvm.so
05-23 16:17:07.305: INFO/DEBUG(27):          #01  pc 0005ad18  /system/lib/libdvm.so
05-23 16:17:07.305: INFO/DEBUG(27):          #02  pc 00054a4a  /system/lib/libdvm.so
05-23 16:17:07.315: INFO/DEBUG(27):          #03  pc 00013f58  /system/lib/libdvm.so
05-23 16:17:07.325: INFO/DEBUG(27):          #04  pc 00019888  /system/lib/libdvm.so
05-23 16:17:07.335: INFO/DEBUG(27):          #05  pc 00018d5c  /system/lib/libdvm.so
05-23 16:17:07.335: INFO/DEBUG(27):          #06  pc 0004d6d0  /system/lib/libdvm.so
05-23 16:17:07.345: INFO/DEBUG(27):          #07  pc 0004d702  /system/lib/libdvm.so
05-23 16:17:07.355: INFO/DEBUG(27):          #08  pc 00041c78  /system/lib/libdvm.so
05-23 16:17:07.365: INFO/DEBUG(27):          #09  pc 00010000  /system/lib/libc.so
05-23 16:17:07.365: INFO/DEBUG(27):          #10  pc 0000fad4  /system/lib/libc.so
05-23 16:17:07.375: INFO/DEBUG(27): code around pc:
05-23 16:17:07.385: INFO/DEBUG(27): ad05a7f4 ffff5ae0 fffe57c4 6801b5f8 6a8b1c05 
05-23 16:17:07.385: INFO/DEBUG(27): ad05a804 1c30681e ff5ef7ff 28001c04 6840d018 
05-23 16:17:07.395: INFO/DEBUG(27): ad05a814 d0152800 37101c27 d0112f00 f7ff1c28 
05-23 16:17:07.395: INFO/DEBUG(27): code around lr:
05-23 16:17:07.405: INFO/DEBUG(27): ad05ad0c f7ff1c20 bd10ff7b 6804b510 fd70f7ff 
05-23 16:17:07.405: INFO/DEBUG(27): ad05ad1c 28001c01 f7ffd102 e002f859 f7ff1c20 
05-23 16:17:07.415: INFO/DEBUG(27): ad05ad2c bd10ff6d 4c24b5f0 1c0d1c06 48236a81 
05-23 16:17:07.425: INFO/DEBUG(27): stack:
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d80  43d20870  /dev/ashmem/mspace/dalvik-heap/2 (deleted)
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d84  00000354  
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d88  00000022  
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d8c  ad043693  /system/lib/libdvm.so
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d90  ad07ff50  /system/lib/libdvm.so
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d94  00000024  
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d98  00000354  
05-23 16:17:07.425: INFO/DEBUG(27):     451c0d9c  ad0170ac  /system/lib/libdvm.so
05-23 16:17:07.425: INFO/DEBUG(27):     451c0da0  00000000  
05-23 16:17:07.435: INFO/DEBUG(27):     451c0da4  afe0f2c0  /system/lib/libc.so
05-23 16:17:07.435: INFO/DEBUG(27):     451c0da8  ad080c00  /system/lib/libdvm.so
05-23 16:17:07.435: INFO/DEBUG(27):     451c0dac  00000002  
05-23 16:17:07.435: INFO/DEBUG(27):     451c0db0  00000354  
05-23 16:17:07.445: INFO/DEBUG(27):     451c0db4  43d20870  /dev/ashmem/mspace/dalvik-heap/2 (deleted)
05-23 16:17:07.445: INFO/DEBUG(27):     451c0db8  df002777  
05-23 16:17:07.455: INFO/DEBUG(27):     451c0dbc  e3a070ad  
05-23 16:17:07.455: INFO/DEBUG(27): #00 451c0dc0  00000000  
05-23 16:17:07.455: INFO/DEBUG(27):     451c0dc4  43d1caa0  /dev/ashmem/mspace/dalvik-heap/2 (deleted)
05-23 16:17:07.455: INFO/DEBUG(27):     451c0dc8  451c0e38  
05-23 16:17:07.455: INFO/DEBUG(27):     451c0dcc  451c0e30  
05-23 16:17:07.455: INFO/DEBUG(27):     451c0dd0  4000a958  /dev/ashmem/mspace/dalvik-heap/zygote/0 (deleted)
05-23 16:17:07.455: INFO/DEBUG(27):     451c0dd4  ad05ad1d  /system/lib/libdvm.so
05-23 16:17:07.465: INFO/DEBUG(27): #01 451c0dd8  417a0b5c  /data/dalvik-cache/system@framework@core.jar@classes.dex
05-23 16:17:07.475: INFO/DEBUG(27):     451c0ddc  ad054a4f  /system/lib/libdvm.so

1
ข้อมูลเพิ่มเติม: เป็นไปได้ที่จะสร้างหัวฉีดที่มีอินสแตนซ์จำลองในโมดูลใด ๆ ฉันสร้าง Injector ในเมธอด "test01 ()" สำเร็จแล้ว แต่ถ้าหัวฉีดถูกสร้างขึ้นโดย RoboUnitTestCase แอปจะหยุดทำงาน
Andrey

16
แหล่งที่มา RoboUnitTestCase code.google.com/p/roboguice/source/browse/roboguice/src/main/...บอกว่า "ให้แน่ใจว่าคุณใช้หนึ่งใน @ * คำอธิบายประกอบการทดสอบและการเริ่มต้นชื่อ testcase ของคุณด้วยการทดสอบ ''" แต่การตั้งค่าของคุณ ไม่มีคำอธิบายประกอบ@Beforeและการทดสอบของคุณไม่มีคำอธิบายประกอบ@Test...
Paul D'Ambra

2
ดูเหมือนว่าจะไม่ใช่ปัญหา java โดยตรง ( signal 11 (SIGSEGV), fault addr 0000000) คุณช่วยลองใช้เฟิร์มแวร์เวอร์ชันอื่น (โปรแกรมจำลองหรืออุปกรณ์) ได้ไหม
Pierre-Henri

คุณช่วยให้คำจำกัดความของอินเทอร์เฟซ InterfaceToMock กับเราได้ไหมเพื่อที่เราจะสร้างพฤติกรรมที่แน่นอนของ AndroidMock.createNiceMock
Remco

ถ้า A ดีสำหรับคุณคุณรับได้ไหม Q ยังเปิดอยู่
Glen Best

คำตอบ:


5

ขออภัยหากมีปัญหากับขั้นตอนการตั้งค่าสำหรับ RoboGuice & การทดสอบหน่วยคุณจะได้รับข้อผิดพลาดประเภทนี้ ไม่มีคำตอบสั้น ๆ ที่วิเศษ แต่เป็นชุดของขั้นตอนที่ต้องปฏิบัติตามอย่างแน่นอน

BTW คุณกำลังใช้ RoboGuice 1.1 - AbstractAndroidModule & RoboUnitTest ไม่มีอยู่ใน RoboGuice 2.0 อีกต่อไป RoboGuice 1.1 จะเลิกเพื่อแก้ปัญหาโดยรวมที่ดีที่สุดคือการย้ายไปยังรุ่นใหม่ล่าสุดตามคำแนะนำเหล่านี้การอัพเกรดเป็น 2.0

อย่างไรก็ตามในกรณีที่คุณเชื่อมต่อกับ RoboGuice 1.1 ขั้นตอนต่อไปนี้:

  • อย่ามีไฟล์โค้ด / บิลด์ที่สร้างขึ้นที่ไม่สอดคล้องกันหลังจากปรับโครงสร้าง / เปลี่ยนชื่อแพ็กเกจเป็นต้นลบโค้ดที่สร้างขึ้นทำการ Clean & Build แม้กระทั่งสร้างโปรเจ็กต์ใหม่และคัดลอกไฟล์ต้นฉบับใน
  • มีรหัสแอปของคุณในโครงการเดียว (ขึ้นอยู่กับ RoboGuice, Instrumentation / RoboUnitTestCase / AndroidMock อิสระ) โครงการรหัสแอปของคุณมีอยู่ใน lib: guice-2.0-no_aop.jar และ roboguice-1.1.2.jar
  • มีรหัสทดสอบหน่วยของคุณในโครงการอื่นที่อ้างถึง (RoboGuice อิสระ, Instrumentation / RoboUnitTestCase / AndroidMock อิสระ) คำแนะนำที่นี่ก่อนที่คุณจะเริ่มต้น โครงการรหัสทดสอบของคุณมีอยู่ใน lib: AndroidMockGenerator.jar
  • ในโครงการแอปของคุณคลาส App + โมดูลของคุณจะมีลักษณะดังนี้:

    package com.mypackage;
    
    import android.app.Instrumentation;
    import android.content.Context;
    
    public class MyApplication extends roboguice.application.RoboApplication {
    
    static MyModule myModule;    
    
    // this constructor usually called by app
    public MyApplication(Context context) {
        super();
        attachBaseContext(context);
    }
    // This constructor called by unit tests.  This is unfortunately small amount of 
    // 'abstraction leakage' of unit test needs into app code.
    public MyApplication(Instrumentation instrumentation) {
        super();
        attachBaseContext(instrumentation.getContext());
    }    
    public static void setModule(MyModule module) {
        MyApplication.myModule = module;
    }   
    public static MyModule getModule() {
        return MyApplication.myModule;
    }   
    }
    

    และ

    package com.mypackage;
    
    public class MyModule extends roboguice.config.AbstractAndroidModule {
    // this will be injected
    protected UsefulObject myUsefulInstance;    
    
    public void setUsefulObject(UsefulObject usefulInstance) {
        this.myUsefulInstance = usefulInstance;
    }    
    public UsefulObject getUsefulObject() {
        return this.myUsefulInstance;
    }    
    
    @Override
    protected void configure() {
        bind(UsefulObject.class).toInstance(myUsefulInstance);
    }
    

    }

  • ในโครงการทดสอบของคุณคลาสกรณีทดสอบของคุณมีลักษณะดังนี้:

    import android.test.suitebuilder.annotation.LargeTest;    
    import com.mypackage.MyApplication;    
    import com.mypackage.MyModule;    
    import com.mypackage.UsefulObject;    
     //import com.mypackage.UsefulObjectSimpleImplementation;    
    import android.test.suitebuilder.annotation.MediumTest;    
    import android.test.suitebuilder.annotation.SmallTest;    
    import com.google.android.testing.mocking.AndroidMock;    
    import roboguice.test.RoboUnitTestCase;
    
    public class TestMyModule extends RoboUnitTestCase<MyApplication> {
    
    @Override
    protected void setUp() throws Exception {
        UsefulObject instance = // new UsefulObjectSimpleImplementation(); 
                                AndroidMock.createNiceMock(UsefulObject.class);           
        MyModule mockModule = new MyModule();
        mockModule.setUsefulObject(instance);
        MyApplication.setModule(mockModule);
        super.setUp();
    }
    
    // Make sure you use one of the @*Test annotations AND begin
    // your testcase's name with "test"
    @MediumTest
    public void test01() {
        AndroidMock.expect(MyApplication.getModule().getUsefulObject().
             simpleMethod()).andStubReturn("Hello!");
    }
    

    }

  • ตรวจสอบให้แน่ใจว่าสำหรับโครงการทดสอบไฟล์ AndroidManifest.xml มีรายการต่อไปนี้:

   <instrumentation android:name="android.test.InstrumentationTestRunner"
     android:targetPackage="com.mypackage"
     android:label="Tests for com.mypackage"/>
  • ก่อนดำเนินการทดสอบตรวจสอบให้แน่ใจว่าโปรแกรมจำลองของคุณเริ่มต้นและทำงานได้อย่างมีสุขภาพดีโดยเรียกใช้แอป "Hello World" ที่เรียบง่ายอื่น ๆ ก่อน เมื่อสำเร็จแล้วให้เรียกใช้แอปของคุณ สุดท้ายเรียกใช้โครงการทดสอบของคุณ

ควรทำงานหลังจากนี้. ขอให้โชคดีแจ้งให้เราทราบ!


0

น่าเสียดายที่นี่เป็นข้อบกพร่องใน Android เอง ดูรายงานข้อผิดพลาดที่นี่ VM จะเกิดปัญหาเมื่อพยายามที่จะมองหาคำอธิบายประกอบในหนังสือมอบฉันทะซึ่งเป็นสิ่งที่ AndroidMock ใช้เมื่อเยาะเย้ยอินเตอร์เฟซ

วิธีแก้ปัญหาคือการสร้างในตัวอย่างของคลาสที่ใช้อินเทอร์เฟซดังที่คุณระบุไว้ในคำถามของคุณ คุณสามารถลองสร้างคลาสนามธรรมบริสุทธิ์ที่ใช้อินเทอร์เฟซโดยไม่ต้องใช้วิธีการใด ๆ จากนั้นใช้ AndroidMock เพื่อล้อเลียนคลาสนั้นแทนอินเทอร์เฟซ สิ่งนี้ควรหลีกเลี่ยงการสร้าง Proxy

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