Тестирование с помощью vitest и testing-library не работает: это связано с использованием SFC Script Setup?

Я новичок в Vue и особенно в функциях композиции. Я пытаюсь протестировать компонент, использующий установку скрипта; однако, кажется, что это не работает.

Компонент вот этот:

      <template>
    <el-card class="box-card" body-style="padding: 38px; text-align: center;" v-loading="loading">
        <h3>Login</h3>
        <hr class="container--separator">
        <el-form ref="formRef"
            :model="form"
        >   
            <el-form-item label="Username">
                <el-input v-model="form.username" placeholder="Username"/>
            </el-form-item>
            <el-form-item label="Password">
                <el-input type="password" v-model="form.password" placeholder="Password" />
            </el-form-item>
            <el-button color="#2274A5" v-on:click="submitForm()">Login</el-button>
        </el-form>
    </el-card>
    
</template>

<script lang="ts" setup>
import {reactive, ref} from 'vue'
import { useRouter } from 'vue-router'
import type {FormInstance} from 'element-plus'
import {useMainStore} from "../../stores/index"
import notification from "@/utils/notification"
import type User from "@/types/User"


const formRef = ref<FormInstance>()
const form: User = reactive({
  username: "",
  password: "",
}) 

const router = useRouter()
const loading = ref(false)

const submitForm = (async() => {
    const store = useMainStore()

    if (form.username === "") {
        return notification("The username is empty, please fill the field")
    }

    if (form.password === "") {
        return notification("The password is empty, please fill the field")
    }

    loading.value = true;
    await store.fetchUser(form.username, form.password);
    loading.value = false;
    
    router.push({ name: "home" })
    

}) 

</script>

<style lang="sass" scoped>
    @import "./LoginCard.scss"
</style>

Когда я пытаюсь проверить это:

      import { test } from 'vitest'
import {render, fireEvent} from '@testing-library/vue'
import { useRouter } from 'vue-router'
import LoginCard from '../LoginCard/LoginCard.vue'

test('login works', async () => {     

    render(LoginCard)
    
})

У меня было больше строк, но простое тестирование для рендеринга компонента дает мне эту ошибку.

      TypeError: Cannot read properties of undefined (reading 'deep')
 ❯ Module.withDirectives node_modules/@vue/runtime-core/dist/runtime-core.cjs.js:3720:17
 ❯ Proxy._sfc_render src/components/LoginCard/LoginCard.vue:53:32
     51|     loading.value = false;
     52| 
     53|     router.push({ name: "home" });

Я попытался прокомментировать части компонента, чтобы увидеть, была ли проблема с определенной линией (например, с маршрутизатором), но проблема, похоже, не устранена. Я пытался искать об этом, но я не знаю, что я делаю неправильно, это связано с самим компонентом? Должен ли я изменить способ создания компонента?

1 ответ

У меня была такая же проблема, и я, наконец, смог понять это. Может быть, это поможет вам.

Проблема заключалась в том, что мне пришлось регистрировать глобальные плагины, используемые моим компонентом при вызове функции рендеринга.

Я пытался протестировать компонент, который использовал директиву, зарегистрированную глобальным плагином. В моем случае это была маска, и я использовал директиву во вводе, который был отрендерен где-то глубоко внутри моего компонента, например так:

      <!-- a global directive my component used -->
<input v-maska="myMask" .../>

@vue/test-utils не распознал его автоматически, что и вызвало проблему. Чтобы решить эту проблему, мне пришлось передать используемый плагин в параметре конфигурации функции:

      import Maska from 'maska';

render(MyComponent, {
    global: {
        plugins: [Maska]
    }
})

Тогда проблема исчезла. Вы можете найти больше информации о конфигурация здесь:https://test-utils.vuejs.org/api/#global

Другие вопросы по тегам