如何对独立的 Vue 组合进行单元测试

Rij*_*ijk 4 vue.js jestjs vue-test-utils vue-composition-api

我有一个类似于文档的示例的设置,其中我的组合位于与我的组件不同的文件中,如下所示:

// composition.js
import { onMounted } from '@vue/composition-api';

export default function useComposition() {
  onMounted(() => {
    console.log('Hello, world!');
  });
}
Run Code Online (Sandbox Code Playgroud)
// component.vue
<template>...</template>

<script>
import { createComponent } from '@vue/composition-api';

export default createComponent({
  setup() {
    useComposition();
  }
})
</script>
Run Code Online (Sandbox Code Playgroud)

我想为该composition.js文件编写一个单独的(基于 Jest 的)测试文件。但是,直接调用组合时,Vue 会报错(意料之中):

// composition.test.js

import useComposition from './composition';

describe('Composition', () => {
  it('Works', () => {
    useComposition();
    // Error: [vue-composition-api] "onMounted" get called outside of "setup()"
  });
});
Run Code Online (Sandbox Code Playgroud)

我已经尝试安装一个基于模拟组合 API 的组件以提供一个setup()功能,但在让它工作时遇到了一些麻烦:

import { mount, createLocalVue } from '@vue/test-utils';
import VueCompositionAPI, { createComponent } from '@vue/composition-api';
import useComposition from './composition';

const localVue = createLocalVue();
localVue.use(VueCompositionAPI);

describe('Composition', () => {
    it('Works', () => {
        const mockComponent = createComponent({
            setup() {
                useComposition();

                return h => h('div');
            }
        });

        mount(mockComponent, { localVue });
        // [Vue warn]: Failed to mount component: template or render function not defined.
    });
});
Run Code Online (Sandbox Code Playgroud)

有没有人对如何让它发挥作用有什么好主意,这样我就可以为我的模块化组合编写测试?

Rij*_*ijk 5

该错误[Vue warn]: Failed to mount component: template or render function not defined.是由@vue/composition-api不支持在 setup 函数中返回渲染函数的事实引起的。有了这些知识,我就能够创建一个有效的模拟组件渲染函数。以下似乎按预期工作:

import VueCompositionAPI from '@vue/composition-api';
import { mount, createLocalVue } from '@vue/test-utils';

const localVue = createLocalVue();
localVue.use(VueCompositionAPI);

const mountComposition = (cb: () => any) => {
    return mount(
        {
            setup() {
                return cb();
            },
            render(h) {
                return h('div');
            }
        },
        { localVue }
    );
};

describe('My Test', () => {
    it('Works', () => {
        let value;

        const component = mountComposition(() => {
            value = useComposition();
        });

        expect(value).toBe(true);
    });
});
Run Code Online (Sandbox Code Playgroud)