Question Comment puis-je faire en sorte qu'une méthode renvoie un argument qui lui a été transmis?


Considérez une signature de méthode comme:

public String myFunction(String abc);

Est-ce que Mockito peut aider à retourner la même chaîne que la méthode reçue?


496
2018-04-21 16:11


origine


Réponses:


Vous pouvez créer une réponse dans Mockito. Supposons que nous ayons une interface nommée Application avec une méthode myFunction.

public interface Application {
  public String myFunction(String abc);
}

Voici la méthode de test avec une réponse Mockito:

public void testMyFunction() throws Exception {
  Application mock = mock(Application.class);
  when(mock.myFunction(anyString())).thenAnswer(new Answer<String>() {
    @Override
    public String answer(InvocationOnMock invocation) throws Throwable {
      Object[] args = invocation.getArguments();
      return (String) args[0];
    }
  });

  assertEquals("someString",mock.myFunction("someString"));
  assertEquals("anotherString",mock.myFunction("anotherString"));
}

Depuis Mockito 1.9.5 et Java 8, il est encore plus simple d'utiliser les fonctions lambda:

when(myMock.myFunction(anyString())).thenAnswer(i -> i.getArguments()[0]);


739
2018-04-26 06:51



Si vous avez Mockito 1.9.5 ou supérieur, il existe une nouvelle méthode statique qui peut faire Answer objet pour vous. Vous devez écrire quelque chose comme

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

Ou bien

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

Notez que le returnsFirstArg() la méthode est statique dans le AdditionalAnswers classe, qui est nouveau dans Mockito 1.9.5; vous aurez donc besoin de la bonne importation statique.


422
2017-08-07 06:17



Avec Java 8, il est possible de créer une réponse d'une ligne même avec une ancienne version de Mockito:

when(myMock.myFunction(anyString()).then(i -> i.getArgumentAt(0, String.class));

Bien sûr, ce n'est pas aussi utile que d'utiliser AdditionalAnswers suggéré par David Wallace, mais pourrait être utile si vous voulez transformer l'argument "à la volée".


40
2018-02-04 20:10



J'ai eu un problème très similaire. L'objectif était de se moquer d'un service qui persiste sur les Objets et de pouvoir les renvoyer par leur nom. Le service ressemble à ceci:

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

Le simulateur de service utilise une carte pour stocker les instances 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;
    }
});

Nous pouvons maintenant exécuter nos tests sur ce simulacre. Par exemple:

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

34
2017-07-19 11:14



Avec Java 8, La réponse de Steve peut devenir

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
    invocation -> {
        Object[] args = invocation.getArguments();
        return args[0];
    });

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}

EDIT: encore plus court:

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
        invocation -> invocation.getArgument(0));

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}

25
2017-07-24 04:34



J'utilise quelque chose de similaire (fondamentalement c'est la même approche). Parfois, il est utile qu'un objet fictif renvoie une sortie prédéfinie pour certaines entrées. Cela va comme ceci:

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
               }
           }
       }
       );

5
2017-11-29 14:45



Vous pouvez utiliser verify () en combinaison avec ArgumentCaptor pour assurer l'exécution du test et ArgumentCaptor pour évaluer les arguments:

ArgumentCaptor<String> argument = ArgumentCaptor.forClass(String.class);
verify(mock).myFunction(argument.capture());
assertEquals("the expected value here", argument.getValue());

La valeur de l'argument est évidemment accessible via l'argument.getValue () pour plus de manipulation / vérification / peu importe.


1
2017-10-28 11:45