個人的な趣味で作っているサービスをVue.js + Firebaseという構成で書かれている。
個人サービス程度にテストはいらんだろって意見もあるけども、毎日書く時間が取れない個人サービスの場合、手を入れようとしたときに自分のコードがどうなっているか覚えていないので、むしろテストがある方が非常に開発がしやすい。
Vueのテストまだ書いたことなかったので入門します。Vue.jsで使わているテストフレームワークをざっとぐぐると、mochaかjestって印象、どちらかというとjestの方がモダンな雰囲気だったので、jestにします。
事前準備
まずvue-cliはアップデートが早かった気がするので最新版にしておく。
$ vue --version 3.10.0
アップデートする。
$ npm install -g @vue/cli
もう4系なのかよオイ。
$ vue --version @vue/cli 4.0.4
公式ドキュメントみながらvue-cliを使って新しいプロジェクトを生成する。
$ vue create vue-unit-test-sample
どうやらこの時点でテストフレームワークを選択出来るっぽいがスルーしてしまった。
ちなみにパッケージマネージャはyarnではなくnpmを指定。今どきはyarnじゃなくてnpmに戻ってきているらしいよ。
サーバ起動してみる。
$ cd vue-unit-test-sample $ npm run serve
http://localhost:8080/ を開く。OKだ。

テスト対象の準備
公式ドキュメントに載っているHelloコンポーネントをコピペする。これはusernameに7文字未満の文字列が入力されているとエラーが表示される、という単純なコンポーネント。
Hello.vue
<template>
<div>
<input v-model="username">
<div
v-if="error"
class="error"
>
{{ error }}
</div>
</div>
</template>
<script>
export default {
name: 'Hello',
data () {
return {
username: ''
}
},
computed: {
error () {
return this.username.trim().length < 7
? 'Please enter a longer username'
: ''
}
}
}
</script>
App.vue
<template>
<div id="app">
<img alt="Vue logo" src="./assets/logo.png">
<!-- <HelloWorld msg="Welcome to Your Vue.js App"/> -->
<Hello />
</div>
</template>
<script>
// import HelloWorld from './components/HelloWorld.vue'
import Hello from './components/Hello.vue'
export default {
name: 'app',
components: {
Hello
}
}
</script>
<style>
#app {
font-family: 'Avenir', Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
text-align: center;
color: #2c3e50;
margin-top: 60px;
}
</style>
画面はこんな感じ

一旦jestを使っていないテストコード。
tests/Hello.test.js
import { shallowMount } from '@vue/test-utils'
import Hello from './Hello.vue'
test('Hello', () => {
// コンポーネントを描画します
const wrapper = shallowMount(Hello)
// `username`は空白を除外して7文字未満は許されません
wrapper.setData({ username: ' '.repeat(7) })
// エラーが描画されることをアサートします
expect(wrapper.find('.error').exists()).toBe(true)
// 名前を十分な長さにします
wrapper.setData({ username: 'Lachlan' })
// エラーがなくなったことをアサートします
expect(wrapper.find('.error').exists()).toBe(false)
})
Jestでテストを書く
公式ドキュメントに従ってやっていきます。
Jest を使用した単一ファイルコンポーネントのテスト Vue Test Utils
$ npm install --save-dev jest @vue/test-utils vue-jest babel-jest
package.jsonに追記
// package.json
{
"scripts": {
"test": "jest"
}
// ...
"jest": {
"moduleFileExtensions": [
"js",
"json",
"vue"
],
"transform": {
".*\\.(vue)$": "vue-jest"
"^.+\\.js$": "<rootDir>/node_modules/babel-jest"
}
}
}
テスト実行してみるも上手く行かない。
$ npm run test
...
FAIL src/components/__tests__/Hello.test.js
● Test suite failed to run
Cannot find module 'babel-core'
at Object.<anonymous> (node_modules/vue-jest/lib/compilers/babel-compiler.js:1:15)
.babelrcの設定飛ばしていたので一応やる。
.babelrc
{
"presets": [["env", { "modules": false }]],
"env": {
"test": {
"presets": [["env", { "targets": { "node": "current" } }]]
}
}
}
$ npm install --save-dev @babel/core @babel/preset-env
テスト実行してみたけど謎のエラー。
$ npm run test
...
Cannot find module 'babel-preset-env' from '/Users/atsushiharada/source/vue-unit-test-sample'
- Did you mean "@babel/env"?
.babelrcを書き換えてみる。
{
"presets": ["@babel/preset-env"]
}
再度テスト実行。エラー内容がもとに戻った。
$ npm run test
...
● Test suite failed to run
Cannot find module 'babel-core'
at Object.<anonymous> (node_modules/vue-jest/lib/compilers/babel-compiler.js:1:15)
何やらbridgeなるものが必要っぽい。とりあえずテスト動かしたいので詳細は追わなかった。
nuxtでjestするとbabel-coreを見つけてくれない - Qiita
$ npm install --save-dev babel-core@bridge
テスト実行。
キタ━━━━(゚∀゚)━━━━!!
~/s/vue-unit-test-sample ❯❯❯ npm run test master ✱ ◼
> vue-unit-test-sample@0.1.0 test /Users/atsushiharada/source/vue-unit-test-sample
> jest
PASS src/components/__tests__/Hello.test.js
Hello
✓ run test !!!!!!!!! (3ms)
Test Suites: 1 passed, 1 total
Tests: 1 passed, 1 total
Snapshots: 0 total
Time: 3.852s
Ran all test suites.
次はテストをjestの形式に書き直します。jestでは、テスト対象がおいてあるディレクトリに__tests__というディレクトリを作って、その下にテストコード入れるっぽい?ちょっとキモいけど郷に従います。
Jest は、テスト対象のコードのすぐ隣にtestsディレクトリを作成することを推奨していますが、適切にテストを構造化することは自由です。
参考
Vue meetupでテスト書いている人が少なかったのでオレオレテストを晒してみる Part. 1 - Qiita
$ tree src src ├── App.vue ├── assets │ └── logo.png ├── components │ ├── Hello.vue │ ├── HelloWorld.vue │ └── __tests__ │ └── Hello.test.js └── main.js
__tests__/Hello.test.jsをjestで書き換える。
describe('Hello', () => {
it('半角スペース7文字入力の場合、エラーになること', () => {
const wrapper = shallowMount(Hello)
wrapper.setData({ username: ' '.repeat(7) })
expect(wrapper.find('.error').exists()).toBe(true)
})
it('7文字以上の文字列を入力した場合、エラーにならないことこと', () => {
const wrapper = shallowMount(Hello)
wrapper.setData({ username: '1234567' })
expect(wrapper.find('.error').exists()).toBe(false)
})
})
通ったー!
$ npm run test
> vue-unit-test-sample@0.1.0 test /Users/atsushiharada/source/vue-unit-test-sample
> jest
PASS src/components/__tests__/Hello.test.js
Hello
✓ 半角スペース7文字入力の場合、エラーになること (23ms)
✓ 7文字以上の文字列を入力した場合、エラーにならないことこと (3ms)
Test Suites: 1 passed, 1 total
Tests: 2 passed, 2 total
Snapshots: 0 total
Time: 1.428s
ちょっと手数多かったけどそれほど難しくはない。次に気になるのは、外部APIコールするメソッドなどをコンポーネントに持たせてたりするけど、そのへんどうやってテストするといいんだろう。
そもそも、外部APIコールする部分は別のレイヤーに切り出してモック出来るようにしておくのが良さそうだけど。