기타

[TDD] Test code 접근 방식, Mock, Mockito, BDD

DongDD 2019. 3. 20. 20:51

[TDD] Test code 접근 방식, Mock, Mockito, BDD



Test Code



Test Method 명명 방식


1. Test 대상 method와 이름을 1:1로 매치

- getName()-> test_getName()


2. 1:1로 매치 후 메소드명에 예상 동작 표현

- getName() -> test_getName_isSuccess()


3. 테스트 시나리오에 맞게 정하기 



접근 방식


1. 시나리오식 접근 방법

1)Happy Day 시나리오

- 정상적인 흐름일 때, 동작해야 하는 결과값 선정

2)Blue Day 시나리오

- 예외나 에러 상황에 대한 결과값 선정


2. 삼각측량법

- 곱하기 메소드의 경우 (a*b)와 같은지, a를 b번 더한 것과 같은지 확인하는 방법


3. Edge Case

- 경계 조건의 값들로 테스트하는 방법



Mock



Mock 객체


- 모듈의 겉모양이 실제 모듈과 비슷하도록 만든 가짜 객체

- ‘의존성’ -> 테스트 작성의 어려움 -> 의존성을 단절시키기 위해 Mock 객체 사용


1. 사용 이유

1) 테스트 작성을 위한 환경 구축의 어려움

- 환경 구축을 위한 작업 시간이 많이 필요한 경우

- 특정 모듈을 갖고 있지 않아 테스트 환경 구축이 어려운 경우 

- 타 부서의 협의나 정책이 필요한 경우

2) 테스트가 특정 경우나 순간에 의존적

- 특정 상태를 가상으로 만들어 놓을 수 있음

3) 테스트 시간이 오래 걸릴 때


Test Double


- 원래의 객체를 사용해서 테스트 진행이 어려운 경우, 이를 대신할 수 있게 만들어 주는 객체


1. Dummy Object

- 인스턴스화 될 수 있을 수준으로만 인터페이스를 구현한 객체

- 해당 객체의 기능이 필요없을 때 사용


2. Test Stub

- Dummy object가 실제로 동작하는 것처럼 보이게 만든 객체

- 특정 상태를 가정해서 만듬


3. Fake Object

- 여러 개의 인스턴스를 대표할 수 있는 객체

- 좀 더 복잡한 구현이 들어가있는 객체

- DB에서 가져온 정보처럼 data를 가지고 있음


4. Test Spy

- 특정 메소드 정상 호출 여부 확인

- 보통 Mock Framework 이용

- EasyMock, jMock, Mockito에 기본적으로 포함되어 있음


5. Mock Object

- 행위를 검증하기 위해 사용되는 객체(행위 기반 테스트)

- 보통 test double과 mock object를 같은 의미로 사용함


Mock Framework


- Mock Object를 직접 생성할 필요없게 도와주는 framework

- Mock 객체에 대한 행위도 테스트 케이스에 포함시킬 수 있음


1. Mockito

- Mock Framework 중 하나

1) 장점

- 테스트의 행위와 반응에만 집중해서 테스트 메소드 작성할 수 있게 해줌

- 테스트 스텁을 만드는 것을 쉽게 하고 만드는 것과 검증을 분리

- Mock을 만드는 방법을 단일화

- API가 간단

- 프레임워크가 지원하지 않으면 안되는 코드를 최대한 배제

- 에러추적이 깔끔    


2) 과정

- Mock 생성 -> Stub -> 수행 -> 검증

- Mock 생성

1
2
3
4
5
6
7
8
9
10
11
12
13
@Test
 public void test_getName_isOk() throws Exception{
        @Data
        @AllArgsConstructor
        @NoArgsConstructor
        class Test {
            private int id;
            private String msg;
        }
        Test testMock = Mockito.mock(Test.class);
        System.out.println(testMock.getId());       // -> 0
        System.out.println(testMock.getMsg());      // -> null
    }
cs


- Stub : Mockito.when으로 특정 메소드가 호출 됐을 때, .thenReturn(return값)을 지정함

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@Test
 public void test_getName_isOk() throws Exception{
        @Data
        @AllArgsConstructor
        @NoArgsConstructor
        class Test {
            private int id;
            private String msg;
        }
        Test testMock = Mockito.mock(Test.class);
        System.out.println(testMock.getId());       // -> 0
        System.out.println(testMock.getMsg());      // -> null        
       Mockito.when(testMock.getId()).thenReturn(1);
        Mockito.when(testMock.getMsg()).thenReturn("MockTest");
        System.out.println(testMock.getId());       // -> 1
        System.out.println(testMock.getMsg());      // -> "MockTest" 
    }
 
cs


-> 결과


- 검증 : Mockto.verify를 사용하여 특정 mock에 대해 어떤 조건을 만족하며 method를 호출했는지 검증

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
@Test
 public void test_getName_isOk() throws Exception{
        @Data
        @AllArgsConstructor
        @NoArgsConstructor
        class Test {
            private int id;
            private String msg;
        }
        Test testMock = Mockito.mock(Test.class);
        System.out.println(testMock.getId());       // -> 0
        System.out.println(testMock.getMsg());      // -> null        
       Mockito.when(testMock.getId()).thenReturn(1);
        Mockito.when(testMock.getMsg()).thenReturn("MockTest");
        System.out.println(testMock.getId());       // -> 1
        System.out.println(testMock.getMsg());      // -> "MockTest" 
 
        Mockito.verify(testMock,Mockito.never()).getId();        // 검증
    }
 
cs


-> 결과


- spy 사용 : 일반 객체를 그대로 가져옴

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@Test
 public void test_getName_isOk() throws Exception{
        @Data
        @AllArgsConstructor
        @NoArgsConstructor
        class Test {
            private int id;
            private String msg;
        }
         Test test = new Test(10,"test");
        Test testMock1 = Mockito.spy(test);
        System.out.println(testMock1.getId());      // -> 10
        System.out.println(testMock1.getMsg());     // -> "test"
    }
 
cs


-> 결과


- SMART_NULL 사용 : RETURNS_SMART_NULLS 사용 시, int는 0, string은 "" 값이 들어감

1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Test
 public void test_getName_isOk() throws Exception{
        @Data
        @AllArgsConstructor
        @NoArgsConstructor
        class Test {
            private int id;
            private String msg;
        }
        Test testMock2 = Mockito.mock(Test.class,Mockito.RETURNS_SMART_NULLS);
        System.out.println(testMock2.getId());      // -> 0
        System.out.println(testMock2.getMsg());     // -> ""
    }
 
cs


-> 결과



BDD(Behavior-Driven Development)


- TDD에 대한 시작점, 범위, 테스트 케이스 등에 대한 문제를 해결해 줄 수 있는 방식

- 어플리켕션의 행위 중 가치 있는 기능부터 개발하여 테스트하는 방식

- TDD(Test-Driven Development) : inside-out 방식(안쪽 모듈 -> 사용자에 가까운 부분)

-> unit test 지향

- BDD : outside-in 방식(사용자에 가까운 부분 -> 안쪽 모듈)

-> 사용자 테스트, 회귀 테스트 지향



TDD와의 차이점


1. BDD는 사용자에 좀 더 가까운 고수준의 기능영역을 우선적으로 다룸


2. 무엇을 테스트할 것인가에 초점이 맞추고 있음


3. Test method 작성에 용이한 문장적인 템플릿 제공

- Given(조건), When(기능 수행), Then(예상 결과)


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@Test
 public void test_getName_isOk() throws Exception{
        @Data
        @AllArgsConstructor
        @NoArgsConstructor
        class Test {
            private int id;
            private String msg;
        }
        Test test = BDDMockito.mock(Test.class);
        System.out.println(test.getId());
        BDDMockito.when(test.getId()).thenReturn(10);
        System.out.println(test.getId());
    }
 
cs

- Spring boot에 있는 BDDMockito 사용

- when으로 특정 method 지정 시, 해당 method를 호출 할 때, 해당 값을 return하도록 설정


-> 결과





Reference) https://repo.yona.io/doortts/blog/issues

Reference) "TDD 실천법과 도구"