테스트 코드
유지보수하기 좋은 테스트 코드 작성
tbonelee
2023. 10. 31. 00:04
- 참고 내용 : "테스트 주도 개발 시작하기" (최범균 저) 10장
변수나 필드를 사용해서 기댓값 표현하지 않기
- 변수나 필드를 사용하면 값 파악이 어렵기 때문에 테스트 코드 파악이 어려워짐
두 개 이상 검증하지 않기
- 개별 검증 대상에 대한 집중도를 높이자
정확하게 일치하는 값으로 모의 객체 설정하지 않기
- 특정한 값에 대해서만 모의 객체가 리턴하도록 하면 테스트할 때 넘긴 값의 작은 변화에도 테스트가 깨져버린다.
- 정확한 값으로 호출되는 것이 중요한 테스트가 아니라면 모의 객체가 반응할 값을 너무 한정짓지 말자.
anyString()
등으로 해줘도 무리 없는 경우들이 많을 것이다.
과도하게 구현 검증하지 않기
- 과도하게 내부 구현을 검증하면 구현 변경으로 테스트가 깨질 가능성이 커진다.
- 가능하면 내부 구현보다 실행 결과를 검증하자.
- 하지만 존재하는 코드에 테스트 코드를 추가하는 경우 기능 검증을 위해 구현을 검증할 수 밖에 없는 경우가 종종 생긴다.
- 불가피한 경우에는 그렇게 하지만 나중에는 리팩토링을 통해 구현이 아닌 결과를 검증하도록 수정해나가자.
셋업을 이용해서 중복된 상황을 설정하지 않기
- 코드 파악의 어려움
- 테스트 작성하고 시간이 많은 이후에는 셋업 메소드로 인해 코드 파악이 어려울 수 있다.
- 테스트간 불필요한 종속성
- 서로 다른 테스트가 불필요하게 논리적으로 결합됨으로써 테스트가 깨지기 쉬워진다. (ex. 셋업 메소드 값 수정)
- 개별 메서드는 스스로 온전하게 설명이 되어야 한다.
실행 환경이 다르다고 실패하지 않기
- OS에 따라 경로 시스템이 다를 수 있음에 주의
실행 시점이 다르다고 실패하지 않기
- 실행하는 시간에 의존하는 메소드
- 대안1) 실행 시점에 의존하는 코드는 시간 관련 객체를 인자로 받아서 실행하는 코드로 수정 -> 테스트에서 시간 관련 객체를 고정하여 검증
- 대안2) 별도의 시간 클래스 작성
- 랜덤 값을 사용하는 메소드
- 대안1) 랜덤 값을 생성자/인자로 받도록 수정
- 대안2) 랜덤 값 생성을 다른 객체에 위임하고 대역 사용
테스트에 필요한 값만 설정
- 불필요한 프로퍼티까지 굳이 설정하지 말자
- null이면 안되는 프로퍼티가 많아서 코드가 복잡해진다면 테스트 코드용 팩토리 클래스를 만들어서 사용해보자.
조건부로 검증하지 않기
- 조건부로 검증하는 경우 조건문을 통과하지 않은 경우에 테스트가 실패하지 않는다.
- 조건부로 검증을 해야 한다면 조건문도 assert해서 실패할 테스트는 실패하도록 만들자.
통합 테스트는 필요하지 않은 범위까지 연동하지 않기
@SpringBootTest
어노테이션을 사용하면 모든 스프링 빈을 초기화하므로 테스트 실행 시간이 길어질 수 있음
더 이상 쓸모 없는 테스트 코드
- ex1) 익숙하지 않은 클래스를 숙지하기 위해 작성해본 테스트 코드
- ex2) 단순히 테스트 커버리지를 높이기 위해 작성한 getter, setter 검증