Vuex 测试


我们主要针对Vuex中的mutaions 和actions进行单元测试。

测试 Mutations

Mutations 很容易被测试,因为它们仅仅是一些完全依赖参数的函数。小技巧是,如果你在 store.js 文件中定义了 mutations,并且使用 ES2015 模块功能默认输出,那么你仍然可以给 mutations 取个变量名然后把它输出去:

  1. const state = { ... }
  2. //取个变量名并输出mutations
  3. export const mutations = { ... }
  4. export default new Vuex.Store({
  5. state,
  6. mutations
  7. })

以下为使用 Mocha + chai 测试 mutation 的例子(实际上你可以用任何你喜欢测试框架)

  1. // mutations.js
  2. export const mutations = {
  3. increment: state => state.count++
  4. }
  1. // mutations.spec.js
  2. import { expect } from 'chai'
  3. import { mutations } from './store'
  4. // 解构赋值mutations(destructure assign mutations)
  5. const { increment } = mutations
  6. describe('mutations', () => {
  7. it('INCREMENT', () => {
  8. // mock state
  9. const state = { count: 0 }
  10. // apply mutation
  11. increment(state)
  12. // assert result
  13. expect(state.count).to.equal(1)
  14. })
  15. })

测试 Actions

Actions 可能会更加棘手一些,因为他们可能要求请求外部API.

当测试actions时,我们通常需要增加mocking服务层——例如,我们可以把API调用抽象成服务,然后我们在测试中模拟这种服务。为了便于解决mock的依赖关系,可以用 Webpack 和 inject-loader 打包测试文件。

异步action测试示例:

  1. // actions.js
  2. import shop from '../api/shop'
  3. export const getAllProducts = ({ dispatch }) => {
  4. dispatch('REQUEST_PRODUCTS')
  5. shop.getProducts(products => {
  6. dispatch('RECEIVE_PRODUCTS', products)
  7. })
  8. }
  1. // actions.spec.js
  2. // 使用 require 语法处理内联loaders
  3. //inject-loader,返回一个模块工厂
  4. //让我们能够注入mocked的依赖关系。
  5. import { expect } from 'chai'
  6. const actionsInjector = require('inject!./actions')
  7. //使用mocks创建模块
  8. const actions = actionsInjector({
  9. '../api/shop': {
  10. getProducts (cb) {
  11. setTimeout(() => {
  12. cb([ /* mocked response */ ])
  13. }, 100)
  14. }
  15. }
  16. })
  17. //用指定的mutatios测试action的辅助函数
  18. const testAction = (action, args, state, expectedMutations, done) => {
  19. let count = 0
  20. // mock 提交
  21. const commit = (type, payload) => {
  22. const mutation = expectedMutations[count]
  23. expect(mutation.type).to.equal(type)
  24. if (payload) {
  25. expect(mutation.payload).to.deep.equal(payload)
  26. }
  27. count++
  28. if (count >= expectedMutations.length) {
  29. done()
  30. }
  31. }
  32. // 用模拟的 store 和参数调用 action
  33. action({ commit, state }, ...args)
  34. // 检查是否没有 mutation 被 dispatch
  35. if (expectedMutations.length === 0) {
  36. expect(count).to.equal(0)
  37. done()
  38. }
  39. }
  40. describe('actions', () => {
  41. it('getAllProducts', done => {
  42. testAction(actions.getAllProducts, [], {}, [
  43. { type: 'REQUEST_PRODUCTS' },
  44. { type: 'RECEIVE_PRODUCTS', payload: { /* mocked response */ } }
  45. ], done)
  46. })
  47. })

测试 Getters

如果你的getters方法很复杂,那么你得测试他们。测试Getter 方法和测试mutations一样非常简单!

测试getter实例:

  1. // getters.js
  2. export const getters = {
  3. filteredProducts (state, { filterCategory }) {
  4. return state.products.filter(product => {
  5. return product.category === filterCategory
  6. })
  7. }
  8. }
  1. // getters.spec.js
  2. import { expect } from 'chai'
  3. import { getters } from './getters'
  4. describe('getters', () => {
  5. it('filteredProducts', () => {
  6. // mock 状态
  7. const state = {
  8. products: [
  9. { id: 1, title: 'Apple', category: 'fruit' },
  10. { id: 2, title: 'Orange', category: 'fruit' },
  11. { id: 3, title: 'Carrot', category: 'vegetable' }
  12. ]
  13. }
  14. // mock getter
  15. const filterCategory = 'fruit'
  16. // 从getter中取回值
  17. const result = getters.filteredProducts(state, { filterCategory })
  18. // 声明返回值
  19. expect(result).to.deep.equal([
  20. { id: 1, title: 'Apple', category: 'fruit' },
  21. { id: 2, title: 'Orange', category: 'fruit' }
  22. ])
  23. })
  24. })

运行测试

如果你的 mutations 和 actions 已经正确,后面应该在适合的mocking上浏览器测试API的依赖关系。

在Node上运行

创建下面的webpack配置(加上适合.babelrc):

  1. // webpack.config.js
  2. module.exports = {
  3. entry: './test.js',
  4. output: {
  5. path: __dirname,
  6. filename: 'test-bundle.js'
  7. },
  8. module: {
  9. loaders: [
  10. {
  11. test: /\.js$/,
  12. loader: 'babel',
  13. exclude: /node_modules/
  14. }
  15. ]
  16. }
  17. }

然后:

  1. webpack
  2. mocha test-bundle.js

在浏览器上运行

  1. 安装 mocha-loader
  2. 把上述 webpack 配置中的 entry 改成 ‘mocha!babel!./test.js’
  3. 用以上配置启动 webpack-dev-server
  4. 访问 localhost:8080/webpack-dev-server/test-bundle.

使用Karma + karma-webpack在浏览器中测试

详细的安装咨询见vue-loader documentation.