mock or stub for chained call

后端 未结 4 1735
不知归路
不知归路 2020-11-29 03:03
protected int parseExpire(CacheContext ctx) throws AttributeDefineException {
    Method targetMethod = ctx.getTargetMethod();
    CacheEnable cacheEnable = targetMe         


        
相关标签:
4条回答
  • 2020-11-29 03:45

    Just in case you are using Kotlin. MockK doesn't say anything about chaining being a bad practice and easily allows you to do this.

    val car = mockk<Car>()
    
    every { car.door(DoorType.FRONT_LEFT).windowState() } returns WindowState.UP
    
    car.door(DoorType.FRONT_LEFT) // returns chained mock for Door
    car.door(DoorType.FRONT_LEFT).windowState() // returns WindowState.UP
    
    verify { car.door(DoorType.FRONT_LEFT).windowState() }
    
    confirmVerified(car)
    
    0 讨论(0)
  • 2020-11-29 03:46

    I found JMockit easier to use ans switched to it completely. See test cases using it:

    https://github.com/ko5tik/andject/blob/master/src/test/java/de/pribluda/android/andject/ViewInjectionTest.java

    Here I mock away Activity base class, which is coming from Android SKD and completely stubbed. With JMockit you can mock thingis tha are final, private, abstract or whatever else.

    In your testcase it would look like:

    public void testFoo(@Mocked final Method targetMethod, 
                        @Mocked  final CacheContext context,
                        @Mocked final  CacheExpire ce) {
        new Expectations() {
           {
               // specify expected sequence of infocations here
    
               context.getTargetMethod(); returns(method);
           }
        };
    
        // call your method
        assertSomething(objectUndertest.cacheExpire(context))
    
    0 讨论(0)
  • 2020-11-29 04:00

    Mockito can handle chained stubs:

    Foo mock = mock(Foo.class, RETURNS_DEEP_STUBS);
    
    // note that we're stubbing a chain of methods here: getBar().getName()
    when(mock.getBar().getName()).thenReturn("deep");
    
    // note that we're chaining method calls: getBar().getName()
    assertEquals("deep", mock.getBar().getName());
    

    AFAIK, the first method in the chain returns a mock, which is set up to return your value on the second chained method call.

    Mockito's authors note that this should only be used for legacy code. A better thing to do otherwise is to push the behavior into your CacheContext and provide any information it needs to do the job itself. The amount of information you're pulling from CacheContext suggests that your class has feature envy.

    0 讨论(0)
  • 2020-11-29 04:02

    My suggestion to make your test case simpler is to refactor your method.

    Anytime I find myself having trouble testing a method, it's a code smell for me, and I ask why is it hard to test. And if code is hard to test, it's probably hard to use and maintain.

    In this case it's because you have a method chain that goes several levels deep. Perhaps pass in ctx, cacheEnable, and cacheExpire as parameters.

    0 讨论(0)
提交回复
热议问题