Les tests vous aident à écrire et maintenir votre code et vous protègent contre les régressions. Les
frameworks de test vous aident à ça, vous permettant de décrire des assertions ou des attentes sur
le comportement de votre code. Svelte ne vous oriente pas sur tel ou tel framework à utiliser — vous
pouvez écrire des tests unitaires, des tests d'intégration, des tests end-to-end en utilisant des
solutions comme [Vitest](https://vitest.dev/), [Jasmine](https://jasmine.github.io/),
[Cypress](https://www.cypress.io/) et [Playwright](https://playwright.dev/)
## Tests unitaires et d'intégration avec Vitest [!VO]Unit and integration testing using Vitest
Les tests unitaires vous permettent de tester des petites parties isolées de votre code. Les tests
d'intégration vous permettent de vérifier comment des morceaux de votre application fonctionnent
ensemble. Si vous utilisez Vite (notamment via SvelteKit), nous vous recommandons d'utiliser
[Vitest](https://vitest.dev/). Vous pouvez utiliser le CLI de Svelte pour [mettre en place
Vitest](/docs/cli/vitest) soit lors de la création du projet, soit plus tard.
Pour mettre en place Vitest manuellement, commencez par installer Vitest :
```bash
npm install -D vitest
```
Puis ajuster votre fichier `vite.config.js` :
```js
/// file: vite.config.js
import { defineConfig } from +++'vitest/config'+++;
export default defineConfig({
// ...
// Dit à Vitest d'utiliser les points d'entrée `browser` dans les fichiers `package.json`, même si
// Vitest utilise Node
resolve: process.env.VITEST
? {
conditions: ['browser']
}
: undefined
});
```
> [!NOTE] Si charger la version navigateur de tous vos paquets n'est pas possible, parce que vous
> testez également des librairies backend, [vous pourriez avoir également besoin d'une configuration
> d'alias](https://github.com/testing-library/svelte-testing-library/issues/222#issuecomment-1909993331)
Vous pouvez maintenant écrire vos tests unitaires dans des fichiers `.js/.ts` :
```js
/// file: multiplier.svelte.test.js
import { flushSync } from 'svelte';
import { expect, test } from 'vitest';
import { multiplier } from './multiplier.svelte.js';
test('Multiplier', () => {
let double = multiplier(0, 2);
expect(double.value).toEqual(0);
double.set(5);
expect(double.value).toEqual(10);
});
```
```js
/// file: multiplier.svelte.js
/**
* @param {number} initial
* @param {number} k
*/
export function multiplier(initial, k) {
let count = $state(initial);
return {
get value() {
return count * k;
},
/** @param {number} c */
set: (c) => {
count = c;
}
};
}
```
### Utiliser les runes dans vos fichiers de test [!VO]Using runes inside your test files
Puisque Vitest traite vos fichiers de test de la même façon que vos fichiers source, vous pouvez
utiliser des runes au sein de vos tests tant que leur nom de fichier inclut `.svelte` :
```js
/// file: multiplier.svelte.test.js
import { flushSync } from 'svelte';
import { expect, test } from 'vitest';
import { multiplier } from './multiplier.svelte.js';
test('Multiplier', () => {
let count = $state(0);
let double = multiplier(() => count, 2);
expect(double.value).toEqual(0);
count = 5;
expect(double.value).toEqual(10);
});
```
```js
/// file: multiplier.svelte.js
/**
* @param {() => number} getCount
* @param {number} k
*/
export function multiplier(getCount, k) {
return {
get value() {
return getCount() * k;
}
};
}
```
Si le code testé utilise des effets, vous aurez besoin de placer le test dans un `$effect.root` :
```js
/// file: logger.svelte.test.js
import { flushSync } from 'svelte';
import { expect, test } from 'vitest';
import { logger } from './logger.svelte.js';
test('Effet', () => {
const cleanup = $effect.root(() => {
let count = $state(0);
// logger utilise un $effect pour afficher les mises à jour de son input
let log = logger(() => count);
// les effets sont en général exécutés après une micro-tâche,
// utiliser `flushSync` pour exécuter tous les effets en attente de manière synchrone
flushSync();
expect(log.value).toEqual([0]);
count = 1;
flushSync();
expect(log.value).toEqual([0, 1]);
});
cleanup();
});
```
```js
/// file: logger.svelte.js
/**
* @param {() => any} getValue
*/
export function logger(getValue) {
/** @type {any[]} */
let log = $state([]);
$effect(() => {
log.push(getValue());
});
return {
get value() {
return log;
}
};
}
```
### Tests de composants [!VO]Component testing
Il est possible de tester vos composants en isolation en utilisant Vitest.
> [!NOTE] Avant d'écrire des tests de composant, posez-vous la question de si vous avez réellement
> besoin de tester le composant, ou si vous souhaitez plutôt tester la logique _au sein_ de votre
> composant. Si c'est le cas, envisagez d'extraire cette logique afin de pouvoir la tester
> indépendamment du composant.
Pour commencer, installer jsdom (une librairie qui simule les APIs du DOM) :
```bash
npm install -D jsdom
```
Puis ajustez votre `vite.config.js` :
```js
/// file: vite.config.js
import { defineConfig } from 'vitest/config';
export default defineConfig({
plugins: [
/* ... */
],
test: {
// Si vous testez des composants côté client, vous aurez besoin de mettre en place un
// environnement DOM. Si tous vos fichiers ne sont pas compatibles avec cet environnement, vous
// pouvez plutôt ajouter un commentaire `// @vitest-environment jsdom` en haut des fichiers de
// test.
environment: 'jsdom'
},
// Dit à Vitest d'utiliser les points d'entrée `browser` dans les fichiers `package.json`, même si
// Vitest utilise Node
resolve: process.env.VITEST
? {
conditions: ['browser']
}
: undefined
});
```
Vous pouvez ensuite créer un fichier de test dans lequel importer le composant à tester, interagir
avec lui programmatiquement et définir les résultats attendus :
```js
/// file: component.test.js
import { flushSync, mount, unmount } from 'svelte';
import { expect, test } from 'vitest';
import Component from './Component.svelte';
test('Component', () => {
// Instantier le composant en utilisant l'API Svelte `mount`
const component = mount(Component, {
target: document.body, // `document` existe grâce à jsdom
props: { initial: 0 }
});
expect(document.body.innerHTML).toBe('');
// Clic sur le bouton, puis synchronisation des changements pour définir les attentes de manière
// synchrone
document.body.querySelector('button').click();
flushSync();
expect(document.body.innerHTML).toBe('');
// Suppression du composant du DOM
unmount(component);
});
```
Même le processus est plutôt simple à mettre en place, celui-ci est aussi bas niveau et plutôt
fragile, puisque la structure de composant peut beaucoup évoluer. Des outils comme
[@testing-library/svelte](https://testing-library.com/docs/svelte-testing-library/intro/) peuvent
aider à industrialiser l'écriture de vos tests. Le test ci-dessus peut ainsi être ré-écrit comme
ceci :
```js
/// file: component.test.js
import { render, screen } from '@testing-library/svelte';
import userEvent from '@testing-library/user-event';
import { expect, test } from 'vitest';
import Component from './Component.svelte';
test('Component', async () => {
const user = userEvent.setup();
render(Component);
const button = screen.getByRole('button');
expect(button).toHaveTextContent(0);
await user.click(button);
expect(button).toHaveTextContent(1);
});
```
Lorsque vous écrivez des tests de composant qui impliquent des liaisons à double sens, du contexte
ou des props de snippet, il est recommandé de créer un composant parent spécifiquement pour votre
test, et interagir avec ce composant. `@testing-library/svelte` montre quelques [exemples de cette
technique](https://testing-library.com/docs/svelte-testing-library/example).
## Tests E2E avec Playwright [!VO]E2E tests using Playwright
Les tests E2E ("end to end" en anglais, qui se traduit par "de bout en bout") vous permettent de
tester votre application toute entière depuis le point de vue de vos utilisateurs. Cette section
prend [Playwright](https://playwright.dev/) comme exemple, mais vous pouvez aussi utiliser d'autres
solutions comme [Cypress](https://www.cypress.io/) ou [NightwatchJS](https://nightwatchjs.org/).
Vous pouvez utiliser le CLI de Svelte pour [mettre en place Playwright](/docs/cli/playwright) soit
pendant la création du projet, soit plus tard. Vous pouvez aussi le [mettre en place avec `npm init
playwright`](https://playwright.dev/docs/intro). De plus, vous pourriez également vouloir installer
un plugin d'IDE tel que l'[extension VSCode](https://playwright.dev/docs/getting-started-vscode)
pour être capable d'exécuter des tests depuis votre IDE.
Si vous avez lancé `npm init playwright` ou n'utilisez pas Vite, vous pourriez avoir besoin
d'ajuster la configuration de Playwright pour lui dire quoi faire avant de lancer les tests —
principalement lancer votre application sur un certain port. Par exemple :
```js
/// file: playwright.config.js
const config = {
webServer: {
command: 'npm run build && npm run preview',
port: 4173
},
testDir: 'tests',
testMatch: /(.+\.)?(test|spec)\.[jt]s/
};
export default config;
```
Vous pouvez maintenant commencer à écrire vos tests. Ils n'ont aucune conscience que Svelte existe
en tant que framework, il vous faudra donc surtout interagir avec le DOM, et écrire vos assertions.
```js
// @errors: 2307 7031
/// file: tests/hello-world.spec.js
import { expect, test } from '@playwright/test';
test("la page d'accueil a le h1 prévu", async ({ page }) => {
await page.goto('/');
await expect(page.locator('h1')).toBeVisible();
});
```