30

I got a class using a factory for creating some object. In my unit test I would like to access the return value of the factory. Since the factory is directly passed to the class and no getter for the created object is provided I need to intercept returning the object from the factory.

RealFactory factory     = new RealFactory();
RealFactory spy         = spy(factory);
TestedClass testedClass = new TestedClass(factory);

// At this point I would like to get a reference to the object created
// and returned by the factory.

Is there a possibility to access the return value of the factory? Probably using the spy?
The only way I can see is to mock the factory create method.

Willi Mentzel
  • 27,862
  • 20
  • 113
  • 121
Marc-Christian Schulze
  • 3,154
  • 3
  • 35
  • 45
  • Why should `TestedClass` take in the factory as the dependency. Shouldn't it just ask for the actual class created by the factory. ?(Law of Demeter) – Ajay George Aug 17 '11 at 16:12
  • The `TestedClass` is an OSGi-Component. A method of the component requires each invocation a new object created by the factory. I refactored the object creation out into a factory class to provide better testability. Since the created object gets initialized depending on the method parameters, there is no way to simply pass in the created objects instead of the factory. – Marc-Christian Schulze Aug 17 '11 at 16:37

2 Answers2

77

First thing, you should be passing spy in as the constructor argument.

That aside, here's how you could do it.

public class ResultCaptor<T> implements Answer {
    private T result = null;
    public T getResult() {
        return result;
    }

    @Override
    public T answer(InvocationOnMock invocationOnMock) throws Throwable {
        result = (T) invocationOnMock.callRealMethod();
        return result;
    }
}

Intended usage:

RealFactory factory     = new RealFactory();
RealFactory spy         = spy(factory);
TestedClass testedClass = new TestedClass(spy);

// At this point I would like to get a reference to the object created
// and returned by the factory.


// let's capture the return values from spy.create()
ResultCaptor<RealThing> resultCaptor = new ResultCaptor<>();
doAnswer(resultCaptor).when(spy).create();

// do something that will trigger a call to the factory
testedClass.doSomething();

// validate the return object
assertThat(resultCaptor.getResult())
        .isNotNull()
        .isInstanceOf(RealThing.class);
Jeff Fairley
  • 8,071
  • 7
  • 46
  • 55
4

The standard mocking approach would be to:

  1. Pre-create the object you want the factory to return in the test case
  2. Create a mock (or spy) of the factory
  3. Prescribe the mock factory to return your pre-created object.

If you really want to have the RealFactory create the object on the fly, you can subclass it and override the factory method to call super.create(...), then save the reference to a field accessible by the test class, and then return the created object.

oksayt
  • 4,333
  • 1
  • 22
  • 41