Cannot access functions in Vue 3 suspense component during unit-test

786 Views Asked by At

I can't access functions of the wrapper.vm in the unit tests once my component become async on the top-level (needed to use <suspense> feature).

<script setup lang="ts">
const state = reactive({ text: '' });

// -------
// How it was before
const addition = '3';  // <-- This works 

// This is what I want!!!!!
const addition = await Promise.resolve('3'); // <-- This "await" doesn't let me access `wrapper.vm.getText()` in unit tests (undefined)
// -------

function getText(): string {
  return '12' + addition;
}

</script>

<template>
  <section class="foo">
    <button @click="getText()">{{ state.text }}</button>
  </section>
</template>

This is how I'm trying to test getText():

describe('some', () => {
  describe('getText', () => {
    it('should return "123"', async () => {
      const wrapper = mount(Some);
      await flushPromises();
      const result = wrapper.vm.getText(); // <--- "getText()" is udnefined!!! 
      expect(result).toBe('123');
    });
  });
});

So, how to access getText() in unit-tests? I use vitest btw.

I know some solutions to test the html, but they aren't letting me to access functions either.

For sure I can test via "click". But what if I need to mock a function?

3

There are 3 best solutions below

1
On

@eXception, Apologize for the late response, your question is quite incomplete though, we usually mention the dev stack like:

  • plugins used like router or store, etc...
  • testing framework

However, I believe;

  • the problem: you just need to access method on wrapper from the test. that said, so depending on your stack you have to await on some process, anyhow
await nextTick(); // is mostly the answer

await router.isReady(); // if you use router

try to check issues/1770#issuecomment-1245751892

Cheers

0
On

For anyone who is looking for workaround

P.S. You might be needed to right configure your plugins and mocks

import { mount, flushPromises } from '@vue/test-utils';
import { describe, expect, it } from 'vitest';
import { defineComponent } from 'vue';
import YourAsyncSetupComponent from '/path/to/your/component';

// Create suspense wrapper for your component
const SuspenseWrapperComponent = defineComponent({
  components: { YourAsyncSetupComponent },
  template: `
    <Suspense>
      <YourAsyncSetupComponent />
    </Suspense>
  `,
});

const createExchangeViewComponent = () => {
  const wrapper = mount(SuspenseWrapperComponent, {
    global: {
      plugins: [
        // set up plugins
      ],
    },
  });

  return wrapper;
};

describe('YourAsyncSetupComponent.vue', () => {
  ///
  ///
  ///
  it('Component renders', async () => {
    // Create and mount suspense wrapper component
    const suspenseWrapper = createExchangeViewComponent();
    // Wait suspense promise
    await flushPromises();

    // Access your target component
    const wrapper = suspenseWrapper.findComponent({ name: 'YourAsyncSetupComponent' });

    // Continue your tests
    expect(wrapper.text()).contains('Your text');

    suspenseWrapper.unmount();
  });
});
1
On

I read the code and (I think) didn't get the point :/

Ideas:

  • refactor:
const getText = async (): Promise<string> => {
  const addition = await Promise.resolve('3');
  return `12${addition}`;
};
  • in the test: await wrapper.vm.getText()

Hope that helps!