[Java] Come posso rendere un metodo restituire un argomento che è stato passato ad esso?


Answers

Se hai Mockito 1.9.5 o versioni successive, c'è un nuovo metodo statico che può rendere l'oggetto Answer per te. Devi scrivere qualcosa del genere

when(myMock.myFunction(anyString())).then(returnsFirstArg());

o in alternativa

doAnswer(returnsFirstArg()).when(myMock).myFunction(anyString());

Si noti che il metodo returnsFirstArg() è statico nella classe AdditionalAnswers , che è nuova a Mockito 1.9.5; quindi avrai bisogno della giusta importazione statica.

Question

Prendi in considerazione una firma del metodo come:

public String myFunction(String abc);

Mockito può restituire la stessa stringa ricevuta dal metodo?




Ho avuto un problema molto simile. L'obiettivo era quello di prendere in giro un servizio che persiste Oggetti e può restituirli con il loro nome. Il servizio si presenta così:

public class RoomService {
    public Room findByName(String roomName) {...}
    public void persist(Room room) {...}
}

Il servizio fittizio utilizza una mappa per memorizzare le istanze Room.

RoomService roomService = mock(RoomService.class);
final Map<String, Room> roomMap = new HashMap<String, Room>();

// mock for method persist
doAnswer(new Answer<Void>() {
    @Override
    public Void answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            Room room = (Room) arguments[0];
            roomMap.put(room.getName(), room);
        }
        return null;
    }
}).when(roomService).persist(any(Room.class));

// mock for method findByName
when(roomService.findByName(anyString())).thenAnswer(new Answer<Room>() {
    @Override
    public Room answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            String key = (String) arguments[0];
            if (roomMap.containsKey(key)) {
                return roomMap.get(key);
            }
        }
        return null;
    }
});

Ora possiamo eseguire i nostri test su questa simulazione. Per esempio:

String name = "room";
Room room = new Room(name);
roomService.persist(room);
assertThat(roomService.findByName(name), equalTo(room));
assertNull(roomService.findByName("none"));



Io uso qualcosa di simile (fondamentalmente è lo stesso approccio). A volte è utile avere un output predefinito di un oggetto mock per determinati input. Questo va così:

private Hashtable<InputObject,  OutputObject> table = new Hashtable<InputObject, OutputObject>();
table.put(input1, ouput1);
table.put(input2, ouput2);

...

when(mockObject.method(any(InputObject.class))).thenAnswer(
       new Answer<OutputObject>()
       {
           @Override
           public OutputObject answer(final InvocationOnMock invocation) throws Throwable
           {
               InputObject input = (InputObject) invocation.getArguments()[0];
               if (table.containsKey(input))
               {
                   return table.get(input);
               }
               else
               {
                   return null; // alternatively, you could throw an exception
               }
           }
       }
       );