DEV Community

Mark
Mark

Posted on

Building Vue3 Component Library from Scratch #12 Vitest

Vitest is a high-performance front-end unit testing framework. Its usage is actually similar to Jest, but its performance is much better than Jest. It also provides good ESM support. Moreover, for projects using Vite as a build tool, a benefit is that they can share the same configuration file, vite.config.js. Therefore, this project will use Vitest as the test framework.

Installation

Since the front-end component library we are testing runs on the browser, we need to install happy-dom additionally. We also need to install the test coverage display tool c8.



pnpm add vitest happy-dom c8 -D -w


Enter fullscreen mode Exit fullscreen mode

Configuration

As mentioned above, Vitest can be configured directly in vite.config.ts, so we go to components/vite.config.ts to make a related configuration for Vitest (the triple-slash directive tells the compiler to include additional files during the compilation process).



/// <reference types="vitest" />
import { defineConfig } from "vite";
import vue from "@vitejs/plugin-vue"
...
export default defineConfig(
    {
        ...
        test: {
            environment: "happy-dom"
        },

    }
)


Enter fullscreen mode Exit fullscreen mode

Then we add two commands in package.json, 'vitest' and 'vitest run --coverage', which are for performing unit tests and viewing unit test coverage respectively.



 "scripts": {
    "test": "vitest",
    "coverage": "vitest run --coverage"
  }


Enter fullscreen mode Exit fullscreen mode

Now we can use Vitest for testing. When executing the 'test' command, Vitest will execute files with the pattern '/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'. Here, we uniformly name our test files in the form of '/*.{test}.ts' and place them in the 'tests' directory of each component.

For example, create a new directory 'tests/button.test.ts' under the 'button' directory, and then write a simple test code to see how it works.



import { describe, expect, it } from 'vitest';

describe('hellostellarnovaui', () => {
  it('should be hellostellarnovaui', () => {
    expect('hello' + 'stellarnovaui').toBe('hellostellarnovaui');
  });
});


Enter fullscreen mode Exit fullscreen mode

Then, by executing 'pnpm run test' in the 'components' directory, we can see that our test has passed.

Image description

Then, by executing 'pnpm run coverage', we can see our test coverage.

Image description

The meaning of each field is as follows:

  • '%stmts' is statement coverage: Has every statement been executed?

  • '%Branch' is branch coverage: Has every 'if' code block been executed?

  • '%Funcs' is function coverage: Has every function been called?

  • '%Lines' is line coverage: Has every line been executed?

How to Test Components?

Above, we just simply tested a string addition, but what we need to test is a component library. So, how do we test whether our components meet the requirements?

Since our project is a Vue component library, we can install the test library recommended by Vue, @vue/test-utils.



pnpm add @vue/test-utils -D -w


Enter fullscreen mode Exit fullscreen mode

Then we modify button.test.ts, we are going to test the slot of the Button component.



import { describe, expect, it } from 'vitest';

import { mount } from '@vue/test-utils';
import button from '../button.vue';
// The component to test
describe('test button', () => {
  it('should render slot', () => {
    const wrapper = mount(button, {
      slots: {
        default: 'stellarnovaui'
      }
    });

    // Assert the rendered text of the component
    expect(wrapper.text()).toContain('stellarnovaui');
  });
});


Enter fullscreen mode Exit fullscreen mode

@vue/test-utils provides a 'mount' method, we can pass in different parameters to test whether the component meets our expectations. For example, the meaning of the above test code is: pass in the button component, and set its default slot to 'stellarnovaui'. We expect the text 'stellarnovaui' to be displayed when the page is loaded. Obviously, our button component has this feature, so when we execute 'pnpm run test', this test will pass.

Image description

If we want to test how the Button component displays a certain style when a type is passed in, we can write it like this.



import { describe, expect, it } from 'vitest';

import { mount } from '@vue/test-utils';
import button from '../button.vue';
// The component to test
describe('test button', () => {
  it('should render slot', () => {
    const wrapper = mount(button, {
      slots: {
        default: 'stellarnovaui'
      }
    });

    // Assert the rendered text of the component
    expect(wrapper.text()).toContain('stellarnovaui');
  });
  it('should have class', () => {
    const wrapper = mount(button, {
      props: {
        type: 'primary'
      }
    });
    expect(wrapper.classes()).toContain('sn-button--primary');
  });
});


Enter fullscreen mode Exit fullscreen mode

The meaning of this test is: when the type we pass in is 'primary', we expect the class name of the component to be 'sn-button--primary'. Obviously, this one can also pass. At the same time, you will find that the test we just started has automatically updated itself, indicating that Vitest has a hot update feature.

Image description

For those interested in more features of @vue/test-utils, you can click on @vue/test-utils to view the official documentation.

The final source code: https://github.com/markliu2013/StellarNovaUI

Top comments (0)