Łapanie wyjątku JUnit

0

Witam chciałbym złapać wyjątek w teście. Jestem początkujący w testach i niebardzo wiem jak to zrobić. Byłbym bardzo wdzięczny za pomoc.

Metoda która tworzy liste userów z pliku:

public static List<User> createUsersList(File users)  {

        List<User> usersList = new ArrayList<>();
        StringBuilder reading = new StringBuilder();
        try {
            Scanner scan = new Scanner(users);
            while (scan.hasNextLine()) {
                String line = scan.nextLine();
                String[] lineArray = line.split(",");
                DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-d");
                LocalDate date = LocalDate.parse(lineArray[2], formatter);


                if (lineArray.length < 4) {
                    usersList.add(new User(lineArray[0], lineArray[1], date, null));
                } else {
                    Long phoneNumber = Long.parseLong(lineArray[3]);
                    usersList.add(new User(lineArray[0], lineArray[1], date, phoneNumber));
                }
            }
        } catch (FileNotFoundException e) {
            System.out.print("Brak pliku o podanej nazwie");

        }

        return usersList;

próba testu :

    @Test
    void createUsersList() throws FileNotFoundException {
        File usersTest = new File("user.txt");
        List<User> usersList=CheckUser.createUsersList(usersTest);
        assertThrows(FileNotFoundException.class,()->{});
    }

1
  1. Wywołanie metody która ma rzucić wyjątek wsadź do lambdy w assertThrows.
  2. Twój test nie przejdzie bo przecież łapiesz wyjątek w bloku catch.
0

czyli coś takiego ?

@Test
    void createUsersList() throws FileNotFoundException {
        File usersTest = new File("user.txt");
        List<User> usersList=CheckUser.createUsersList(usersTest);
        assertThrows(FileNotFoundException.class,()->
        {CheckUser.createUsersList(usersTest);
        
        });

niestety cały czas nie łapie

0

Twoja metoda nie rzuci wyjątkiem tylko go obsłuży. To znaczy, ze mógłbyś sprawdzić sobie wyjście consoli. https://stackoverflow.com/questions/1119385/junit-test-for-system-out-println

2

@witu122 tak się nie da, bo twoja metoda NIE RZUCA wyjątku! Co jest zresztą dramatycznie źle napisane, bo teraz ktoś zawoła metodę createUsersList i może dostać pustą listę, myśląc że załadował dane z pliku.

  1. Albo rzuć ten wyjątek z metody i pozbądź się catch
  2. Albo lepiej zlap wyjątek a z metody zwracaj Optional<List<User>>> sygnalizując userowi że ta metoda może failować
  3. Albo jeszcze lepiej zlap wyjątek a z metody zwracaj Either<Exception, List<User>>> sygnalizując userowi że ta metoda może failować i zwracając w takiej sytuacji informacje o błędzie

Na marginesie logowanie w stylu System.out.print("Brak pliku o podanej nazwie"); to też dramat, bo co jak odpalisz to gdzieś bez wyjścia konsoli? Używaj Loggera ;)

0

@Shalom: szczerze powiedziawszy tak jak mówiłem jestem jeszcze troche zielony w tych testach, czy mógłbyś mi jakoś łopatologicznie przedstawić pkt1 lub 2 albo napisać pseudokod ?

1

Twoja metoda ŁAPIE wyjątek i go obsługuje, więc nigdy go nie rzuci. To jak w teście chcesz sprawdzić czy metoda rzuca wyjątek?

  1. Usuwasz try catcha, a dodajesz w metodzie throwsa.
  2. Tutaj zostawiasz try catcha, ale opakowujesz liste w optionala. Potem w teście sprawdzasz czy optional jest pusty czy nie.

Myślałem, żę @Shalom wyjaśnił to w miarę jasno ;)

4

@witu122:

szczerze powiedziawszy tak jak mówiłem jestem jeszcze troche zielony w tych testach

Twój problem polega na nieznajomości Javy a nie testów.

W twoim kodzie łapiesz wyjątek, wiec jeśli pliku nie ma, to na konsole wypiszesz System.out.print("Brak pliku o podanej nazwie"); i zwrócisz userowi pustą listę. Jeśli NIE chcesz żeby tak było, to wywal tego try-catcha całego i zrób:

public static List<User> createUsersList(File users)  throws FileNotFoundException {
	List<User> usersList = new ArrayList<>();
	StringBuilder reading = new StringBuilder();
	Scanner scan = new Scanner(users);
	while (scan.hasNextLine()) {
		String line = scan.nextLine();
		String[] lineArray = line.split(",");
		DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-d");
		LocalDate date = LocalDate.parse(lineArray[2], formatter);

		if (lineArray.length < 4) {
			usersList.add(new User(lineArray[0], lineArray[1], date, null));
		} else {
			Long phoneNumber = Long.parseLong(lineArray[3]);
			usersList.add(new User(lineArray[0], lineArray[1], date, phoneNumber));
		}
	}
	return usersList;
}

Wtedy ten twój test przynajmniej zacznie działać, bo teraz metoda faktycznie rzuca wyjątek. Ale to nieładne i utrudnia pisanie ładnego kodu, więc możemy zrobić inaczej:

public static Optional<List<User>> createUsersList(File users)  {

	List<User> usersList = new ArrayList<>();
	StringBuilder reading = new StringBuilder();
	try {
              // bla bla bla
	} catch (FileNotFoundException e) {
		LOGGER.error("Brak pliku o podanej nazwie",e);
		return Optional.empty();
	}

	return Optional.of(usersList);
}

Teraz w teście zamiast assertThrows sprawdzałbyś po prostu czy wynik jest empty czy nie. Minus jest taki, ze user który wywoła tą metodę nie dostanie informacji co było błędem. Tutaj to bez różnicy, bo pewnie błąd może być tylko jeden, ale czasem chcesz takie rzeczy wiedzieć. Można by w takim razie zrobić:

public static Either<Exception, List<User>> createUsersList(File users)  {

	List<User> usersList = new ArrayList<>();
	StringBuilder reading = new StringBuilder();
	try {
             // bla bla bla
	} catch (FileNotFoundException e) {
		LOGGER.error("Brak pliku o podanej nazwie",e);
		return Either.left(e);
	}

	return Either.right(usersList);
}

Teraz w teście trzeba by sprawdzać czy wynik jest left czy right.

Zastanowiłbym się też, czy ta metoda serio powinna być static, bo znów to jest często spory ból jak musisz coś testować a tam pod spodem latają statici, bo nagle nie bardzo jest jak to "nadpisać" testową implementacją.

1

Tak poza konkursem, to nie wolałbyś rozbić tej metody na mniejsze części? np.
a) List<String> getUserDescriptorFromFile(File users);
b) List<xxx> createUsers(List<String> csvData);

1 użytkowników online, w tym zalogowanych: 0, gości: 1