Vue testing handbook
이 글은 원작자의 허락하에 Vue testing handbook의 일부를 번역한 글입니다.
번역문에 대한 좋은 의견이 있으면 말씀해주세요.
감사합니다.
이벤트 트리거 하기
Vue 컴포넌트에서 하는 일 중 가장 일반적인 것은 사용자의 입력(input)을 받는 일입니다. vue-test-utils
와 Jest는 입력을 테스트하는 일을 쉽게 만들어줍니다. trigger
와 컴포넌트가 정확하게 작동하는지 검증하는 Jest mocks를 어떻게 사용하는지 살펴보겠습니다.
이 페이지에 설명한 테스트의 소스 코드는 여기서 찾을 수 있습니다.
컴포넌트 생성하기
간단한 형태의 컴포넌트를 만들어보겠습니다. <FormSubmitter>
라는 이름이며, <input>
과 <button>
을 가지고 있습니다. 버튼을 클릭했을 때, 어떤 일이 발생합니다. 첫 예제는 간단하게 성공 메시지를 드러내겠습니다. 그러고 나서 외부 엔드포인트(endpoint)에서 form을 보내는 좀 더 흥미로운 예제를 살펴보겠습니다.
<FormSubmitter>
를 생성하고 템플릿 부분으로 갑니다.
<template>
<div>
<form @submit.prevent="handleSubmit">
<input v-model="username" data-username>
<input type="submit"
</form>
<div
class="message"
v-if="submitted"
>
Thank you for your submission, {{ username }}
</div>
</div>
</template>
유저가 form을 제출할 때, 제출해줘서 고맙다는 감사 메시지를 드러냅니다. 여기서 비동기적으로 form을 제출하기를 원합니다. 그래서 디폴트 액션(default action)을 막아주기 위해 @submit.prevent
를 사용하고 있습니다. 디폴트 액션은 form이 제출될 때 페이지를 새로고침(refresh) 해줍니다.
이제 form 제출 로직을 추가하겠습니다.
<script>
export default {
name: "FormSubmitter",
data() {
return {
username: '',
submitted: false
}
},
methods: {
handleSubmit() {
this.submitted = true
}
}
}
</script>
꽤 간단합니다. 단지 form이 제출됐을 때, submitted
를 true
로 설정합니다. 그러고 나면 성공 메시지를 담고 있는 <div>
태그가 드러납니다.
테스트 작성하기
작성한 테스트를 보겠습니다. async
를 이 테스트에 표시했습니다. 코드를 읽어보고 이유를 찾아보세요.
import { shallowMount } from "@vue/test-utils"
import FormSubmitter from "@/components/FormSubmitter.vue"
describe("FormSubmitter", () => {
it("제출했을 때 알림이 나타난다.", async () => {
const wrapper = shallowMount(FormSubmitter)
wrapper.find("[data-username]").setValue("alice")
wrapper.find("form").trigger("submit.prevent")
await wrapper.vm.$nextTick()
expect(wrapper.find(".message").text())
.toBe("Thank you for your submission, alice.")
})
})
테스트는 설명할 필요도 없이 꽤 명백합니다. 컴포넌트를 shallowMount
하고, username을 설정합니다. 그리고 사용자의 입력을 시연하기 위해 제공하는 vue-test-utils
의 trigger
메서드를 사용합니다. trigger
는 커스텀 이벤트로 작동합니다. 수식어(modifier)를 사용하는 이벤트인 submit.prevent
, keydown.enter
같은 것들도 마찬가지입니다.
trigger
를 호출한 후에 주목하세요, await wrapper.vm.$nextTick()
을 호출합니다. async
로 테스트를 표시해야만 하는 이유입니다. 그래야 await
을 사용할 수 있습니다. vue-test-utils
의 beta 28 버전 시점에서, Vue의 반응성 시스템이 DOM을 업데이트한 것을 확인하기 위해서 nextTick
을 호출하는 것이 필요합니다. 때때로 nextTick
을 호출하지 않고 진행할 수도 있습니다. 그러나 만약 컴포넌트가 복잡해지기 시작한다면, 레이스 컨디션(race condition)을 맞이할 수도 있고, 어설션(assertion)이 Vue가 DOM을 업데이트하기 전에 동작할 수도 있습니다. 이 부분에 대해서 더 알고 싶다면 vue-test-utils documentation을 읽어보세요.
위 테스트는 유닛 테스트의 세 단계를 따릅니다.
- 준비 (테스트를 위한 설정을 합니다. 위의 경우에 컴포넌트를 렌더합니다)
- 행동 (시스템에 있는 행동을 실행합니다)
- 검증 (실제 결과가 기대했던 것과 일치하는지 확인합니다)
새로운 라인으로 각각의 단계를 구분해서 테스트를 좀 더 읽기 수월하게 만듭니다.
트리거는 매우 간단합니다. 몇 가지 입력을 시연하기 원하는 엘리먼트를 갖기 위해 find
를 사용합니다. 그리고 이벤트의 이름과 임의의 수식어를 가지고 trigger
를 호출합니다.
실제 사례
Form은 보통 일부 엔드포인트에서 제출됩니다. handleSubmit
의 다른 구현체를 가지고 어떻게 이 컴포넌트를 테스트 할 수 있을지 알아보겠습니다. 한 가지 일반적인 경우는 Vue.prototype.$http
의 HTTP 라이브러리에 가명을 사용하는 것입니다. 이렇게 하면 간단하게 this.$http.get(...)
을 호출해서 ajax 요청(request)를 할 수 있게 해줍니다.
이 일은 ajax 요청(request)를 하게 해줍니다. 간단하게 this.$http.get(...)
을 호출함으로써. 이 예제에 대해서는 여기서 좀 더 배워보세요.
보통 http 라이브러리는 인기있는 HTTP 클라이언트인 axios
입니다. 이 경우에 handleSubmit
은 아래와 같이 보일 것입니다.
handleSubmitAsync() {
return this.$http.get("/api/v1/register", { username: this.username })
.then(() => {
// 성공 메세지나 기타 등등을 보여줍니다
})
.catch(() => {
// 에러를 다룹니다
})
}
이 경우에, 한 가지 테크닉은 원하는 테스트 환경을 조성하기 위해서 this.$http
를 mock 하는 것입니다. mocks
마운팅 옵션에 대해서는 여기서 읽어볼 수 있습니다. http.get
메서드의 mock 구현체를 살펴보겠습니다.
let url = ''
let data = ''
const mockHttp = {
get: (_url, _data) => {
return new Promise((resolve, reject) => {
url = _url
data = data
resolve()
})
}
}
여기서 몇 가지 흥미로운 요소가 있습니다.
$http.get
에 넘겨진url
과data
를 저장하기 위한url
과data
변수를 생성합니다. 이 작업은 요청(request)가 올바른 엔드포인트와 페이로드(payload)에 맞았는지 검증하는 데 유용합니다.url
과data
인자를 할당한 후에, 성공적인 API 응답(response)을 시연하기 위해서 즉시 프로미스(promise)를 resolve 합니다.
테스트를 보기 전에, 새로운 handleSubmitAsync
함수를 보겠습니다.
methods: {
handleSubmitAsync() {
return this.$http.get("/api/v1/register", { username: this.username })
.then(() => {
this.submitted = true
})
.catch((e) => {
throw Error("무언가가 잘못됐습니다.", e)
})
}
}
새로운 handleSubmitAsync
메서드를 사용하기 위해서 <template>
도 업데이트 합니다.
<template>
<div>
<form @submit.prevent="handleSubmitAsync">
<input v-model="username" data-username>
<input type="submit">
</form>
<!-- ... -->
</div>
</template>
이제 테스트 하는 일만 남았습니다.
ajax 호출을 모킹하기
맨 처음, describe
블록 전의 윗 부분에서 this.$http
의 mock 구현체를 추가합니다.
let url = ''
let data = ''
const mockHttp = {
get: (_url, _data) => {
return new Promise((resolve, reject) => {
url = _url
data = _data
resolve()
})
}
}
이제 mocks
마운팅 옵션에 $http
mock을 넘기는 테스트를 추가합니다.
it("제출했을 때 알림이 나타난다.", () => {
const wrapper = shallowMount(FormSubmitter, {
mocks: {
$http: mockHttp
}
})
wrapper.find("[data-username]").setValue("alice")
wrapper.find("form").trigger("submit.prevent")
expect(wrapper.find(".message").text())
.toBe("Thank you for your submission, alice")
})
이제 실제 http 라이브러리를 Vue.prototype.$http
에 부착해서 사용하는 것 대신에, 위의 mock 구현체를 대신 사용하겠습니다. 테스트 환경을 통제하고 일정한 결과를 가질 수 있다는 점에서 이렇게 하는 것이 좋습니다.
yarn test:unit
을 실행하면 실제로 아래와 같이 실패하는 테스트를 산출합니다.
FAIL tests/unit/FormSubmitter.spec.js
● FormSubmitter › reveals a notification when submitted
[vue-test-utils]: find did not return .message, cannot call text() on empty Wrapper
mockHttp
가 반환한 프로미스가 resolve 되기 전에 테스트가 끝나는 일이 일어나고 있습니다. 아래처럼 테스트를 비동기로 만들 수 있습니다.
it("reveals a notification when submitted", async () => {
// ...
})
하지만 테스트는 여전히 프로미스가 resolve 되기 전에 끝날 것입니다. 이것과 관련해서 할 수 있는 한 가지 방법은 flush-promise를 사용하는 것입니다. Flush-promise는 간단한 Node.js 모듈인데, 모든 pending 상태의 프로미스를 즉시 resolve 하는 일을 합니다. yarn add flush-promises
로 라이브러리를 설치하고, 아래와 같이 테스트를 업데이트합니다.
import flushPromises from "flush-promises"
// ...
it("reveals a notification when submitted", async () => {
const wrapper = shallowMount(FormSubmitter, {
mocks: {
$http: mockHttp
}
})
wrapper.find("[data-username]").setValue("alice")
wrapper.find("form").trigger("submit.prevent")
await flushPromises()
expect(wrapper.find(".message").text())
.toBe("Thank you for your submission, alice.")
})
flush-promises
를 사용하면 모든 프로미스를 확인하는 꽤 괜찮은 부수 효과를 가집니다. 이 부수효과에는 nextTick
을 resolve 한 것과 Vue가 DOM을 업데이트한 것을 포함됩니다.
이제 테스트는 통과합니다. flush-promises
의 소스코드는 10줄밖에 안 됩니다. Node.js에 관심이 있다면, 읽어보고 어떻게 작동하는지 이해하는 것도 값진 일일 것입니다.
엔드포인트와 페이로드가 정확한지도 확인해야 합니다. 테스트에 두 개의 어설션을 추가합니다.
// ...
expect(url).toBe("/api/v1/register")
expect(data).toEqual({ username: "alice" })
테스트는 여전히 통과합니다.
결론
이 섹션에서는 아래와 같은 방법을 알게 됐습니다.
prevent
같은 수식어를 사용하는 이벤트도 포함한 이벤트에trigger
를 사용하는 방법v-model
을 사용하는<input>
의 값을 설정하기 위해setValue
사용하는 방법- 유닛 테스트의 세 단계를 사용해서 테스트를 작성하는 방법
mocks
를 마운트하는 옵션을 사용해서,Vue.prototype
에 부착된 메서드를 mock 하는 방법- 모든 프로미스를 즉시 해결하기 위해서, 유닛 테스트에서 유용한 기술인
flush-promises
를 사용하는 방법
이 페이지에 설명한 테스트의 소스 코드는 여기서 찾을 수 있습니다.