Chapter8. 테스트
아주 재밌었다. 앞으로 자주 사용해야겠다.
먼저 unit테스트, Functional 테스트 등 작은 단위의 테스트에 대해 학습을 한다.
이후 실제 사용자의 행동를 모방하고, 테스트 하는 Integration 테스트
를 학습한다.
8.1 테스트
애플리케이션에서 테스트는 빼놓을 수 없는 작업이다.
코드를 작성하고 브라우저를 통해 올바른 결과가 나오는지를 확인하는 것도 좋은 방법이다. 하지만 이런 방법은 자동화가 되지 않고 개발자가 직접 하나하나 확인해야 하므로 효율상 좋지 않다.
그렇다면 어떤 방법이 효율성이 좋은 테스트인가? 당연 자동화 시키는 것.
또한 Rails는 자동화된 테스트를 굉장히 중요시한다. 아래와 같은 테스트를 지원한다.
여기서는 테스트 프레임워크로 Rails의 표준인
Test::Unit
을 살펴볼것임.
8.2 테스트 준비
Rails에서 테스트를 하려면 데이터베이스와 테스트 전용 데이터가 필요하다
8.2.1 테스트 데이터베이스 구축
rake 명령어를 사용
터미널에 다음과 같은 명령어를 실행해보자
8.3 Unit 테스트
8.3.0 Unit 테스트
Unit 테스트(단위 테스트)
는 애플리케이션을 구성하고 있는 라이브러리(주요 모델 등)가 제대로 작동하는지 확인하는 테스트Rails에서 실행하는 테스트 중에서도 가장 기본적인 테스트다.
8.3.1 Unit 테스트 기본
책의
3.7.2
절을 잘 따라했으면/test/models
폴더 내부에 테스트 스크립트로book_test.rb
가 생성되어 있을 것임 -- 어떻게 하면 생기는지 확인해보자book_test.rb
의 첫 모습은 아래와 같다.주석으로 표시된 부분이 테스트를 어떻게 하는지에 대해 간단히 설명해 주는 부분이다.
먼저
test
메서드를 정의해줘야 한다.test
메서드의 매개 변수는test 이름
과&block
이다.test이름
은 말 그대로 어떤 테스트의 이름이고&block
은 테스트에 넘겨줄 실행 블록을 의미한다.
간단한 테스트를 만들어보자. 우리가 만들 테스트는 아래와 같다.
book
레코드를 하나 만든다.book
을save
한다.만약
save
에 실패하면Test
의Fail
아래서 사용된
assert
메서드는 종류가 무진장 많다. 테스트의 종류에 맞춰서 사용하면 된다. 자세한 정보는 여기에서 알아보자.
이후 터미널에서 테스트를 진행하자!
만약
book.save
가 실패한다면 터미널에 실패했다는 표시인F
와 함께assert
의 두 번째 인자로 준msg
가 출력이 된다.테스트 메서드가 여러 개 있을 때는 실행 순서가 마음대로다. 따라서 실행순서에 의존하는 테스트 메서드는 작성하지 말자!
8.3.2 Unit 테스트 구체적인 예제
모델 유효성 결과 확인
유효성 검사 오류 정보는
errors
메서드로 접근할 수 있다!
앞서서 간단한 저장 테스트를 해봤다.
이번에는 모델에서 정의된 유효성 검사의 동작인 확인해보자. 아래와 같이 스크립트를 작성하자 이 스크립트는 고의적으로 생성한
book
이 유효성 검사를 통과하지 못하게 만드는 스크립트다.assert !book.save, 'Failed to Validate!'
유효성 검사를 통과하지 못해, 모델 저장에 실패했을 경우
assert_equal book.errors.size, 2, 'Failed to Validate'
assert_equal
은 첫 번째, 두 번째 인자를 비교해서 만약 값이 같으면true
를 반환하고 아니라면false
를 반환한다.두 개의 유효성 검사가 있을 경우
assert book.erros[:isbn].any?, 'Failed to isbn validate'
isbn 필ㄷ의 윻성 검사 오류가 적어도 한 개 이상 있을 경우
모델을 통해 검색한 결과를 확인
모델을 통해 검색했을 때, 정확한 결과를 추출하는지 확인해보자
Book
모델에서title 필드
를 키로자바스크립트 라이브러리 실전 활용
을 검색했을 때 다음을 확인한다.추출한 결과가 Book 객체인지 확인
isbn 필드가 픽스처
books.yml
에 있는:jslib
키의 isbn 필드와 같은지 확인publish 필드가
2013/03/19
인지 확인
Rails는 테스트를 실행할 때 픽스처를 데이터베이스에 로드하는 것뿐만 아니라, 테스트 스크립트에서 이용할 수 있게 해시로 전개해준다!
따라서
books.yml
내부에:jslib
이라는 키로 정의된 레코드가 있다면,books(:jslib)
에 접근할 수 있다.그리고 모델 객체를 리턴하므로 그대로 isbn 속성에 접근할수도 있다.
이러한 기능덕분에
assert_equal books(:jslib).isbn
같은 코드를 작성할 수 있는것.books.yml
에서jslib
는 아래와 같다.
뷰 헬퍼 테스트 -- 실습실패
뷰 헬퍼 테스트도 기본적인 방법은 모델 테스트와 동일하다.
2.2.1절
에서 컨트롤러를 순서대로 생성했다면,/test/helpers
폴더 내부에
테스트 준비와 뒤처리 - setup과 teardown
테스트 스크립트에는 각각의 테스트 메서드가 호출되기 전과 후에 자동으로 호출되는 메서드가 있따.
테스트 스크립트의 부모 클래스
ActiveSupport::TestCse
에 정의되어 있는 메서드 이므로 오버라이드해서 사용한다.앞서 진행했던
where method test
를setup
과teardown
으로 수정해보자
8.4 Functional 테스트
Gemfile 에
rails-controller-testing
을 추가해주자
Function 테스트
는 컨트롤러의 동작 또는 템플릿의 출력을 확인하기 위한 테스트다.Functional 테스트
에서는 브라우저처럼HTTP
요청을 유사적으로 작성하는 것으로 액션 메서드를 실행하고 그 결과로HTTP
상태코드, 템플릿 변수 또는 최종적인 출력 구조 등을 확인한다.또한 라우트 정의의 유효성을 확인하는 것도
Functional테스트
의 역할이다.
8.4.1 Functional 테스트 기본
/test/controllers
폴더 내부에 테스트 스크립트로 존재한다.hello_controller_test.rb
룰 수정해서hello
컨트롤러를 테스트 하자. 아래처럼 스크립트를 작성하자위 코드를 분석해보자
1. Get 메서드로 요청 생성
Functional 테스트에서는 일단 컨트롤러를 실행하기 위해
get
메서드로HTTP
요청을 생성한다.get
메서드가 아닌POST, PUT
등등도 다 사용할 수 있ㄷ.
2. Functional 테스트에서 이용할 수 있는 예약 변수
Functional
텥스트에서는get, post
등의 메서드를 사용한 후에 아래의 표처럼 예약 변수에 접근할 수 있다.이러한 변수를 통해 액션 메소드의 내부에서 생성된 정보에 접근하자
위의 테스트에서
assert_equal 10, assigns(:books).length, 'found rows is worng.'
부분을 보면assigns(:books)
가 의미하는 바는hello/list
의 템플릿 변수@books
를 추출하는 과정이다. 그리고 이@books
의 사이즈를10
과 비교하는 테스트다.
3. Functional 테스트에서 사용할 수 있는 assert_xxxxx 메서드
assert_response :success, 'list action failed.'
이 코드는 테스트 결과 response가 200인지 확인하는 코드다.assert_template 'hello/list'
는get hello_list_path
의 결과hello/list
뷰 템플릿이 잘 호출되는지 확인한다.
8.4.2 Functional 테스트에서 사용할 수 있는 Assertion 메서드
각 Assertion 메서드를 사용하는 법을 확인해보자
assert_difference 메서드
처리 후의 상태 변화를 확인한다.
assert_difference
메서드는 블록 내부의 처리를 실행했을 때, 값이 변화되는지를 확인하는 Assertion 메서드다.책에 있는 그대로 하면 실행되지 않는다.
post
HTTP메서드를 보낼url
을books_path
로 수정해주자.book
생성을params
으로 감싸준다.
위의 스크립ㅌ는
create
액션을 수행하고 주어진 포트스 데이터를 기반으로 도서 정보 하나를 생성, 추가한다.따라서
create
액션이 정상적으로 완료되면books
테이블의 데이터 개수가 1만큼 증가할 것이다.기존에는 10, create가 성공하면 11
이전의
Books.count
와 성공 이후의Books.count
는 1만큼 차이가 날 것임.만약 실패하면 카운트가 증가하지 않고 그대로 10. 테스트는 실패한다.
만약 변화하지 않는걸 테스트하기 위해서는
assert_no_difference
로 하면 된다.
assert_generates
라우팅 동작 확인
assert_generates
메서드는 두 번째 매개 변수로 지정한 컨트롤러와 액션이 첫 버째 매개 변수에 입력한 경로로 들어갔을 때 실행되는지 확인한다.두 번째 매개 변수에는
url_for
메서드의 매개 변수처럼 내용의 해시를 지정할 수도 있다.위와 같은 스크립트는
localhost:3000/hello/list
에 접근했을 때 아래를 확인한다.hello
컨트롤러의list
액션 메서드가 실행되는지
assert_recognizes
컨트롤러의 액션메서드가 지정한 라우트로 들어가는지 확인한다.
assert_select
템플릿 출력 결과를 확인한다
이 메서드는 뷰의 출력 결과를 확인하기 위한 메서드다. 다양한 테스트를 할 수 있다.
매개 변수
selector
에는CSS
선태자를 지정한다.이 메서드에는 선택자로
HTML
요소에 간단히 접근할 수 있다는 것이 장점아래는
assert_select
메서드에서 사용할 수 있는 선택자들이다홀수줄에 나오는 게 선택자, 의미
짝수줄에 나오는게 바로 위의 선택자, 의미의 예시다.
HTML 문서에서 특정한 요소를 추출하여 이를 어떤 방식으로 테스트할 것인지를 나타내는 것이 매개 변수
equality
이다. 아래는 구체적인 사용 방법이다.지정할 수 있는 값들은 아래에 표를 참고하자
8.5 Integration 테스트
이 테스트는 책의 예시가 아닌 내가 간단하게 만든 사이트로 할것임
Integration 테스트
는 통합 테스트라고도 부른다.여러 개의 컨트롤러를 넘나들며 실제 사용자의 행동을 모방할 때 사용한다.
아래의 순서로 테스트 할 것임.
get new_post_path
로 포스트 생성에 접근로그인이 안되었으면 자동으로
login/index
로 접근해야함로그인 페이지에서 사용자 이름과 비밀번호를 입력하고 인증 처리
로그인에 성공하면
posts_path
로 리다이렉트
위의 스크립트는 3번에서 뭔가 잘못작동한다!
테스트결과 로그인을 실패하고 내가 의도한곳으로
redirect
되지 않는다. 실패원인 찾아볼것!
Last updated