First, enable Babel support in Jest as documented in the Getting Started guide.

Let's implement a simple module that fetches user data from an API and returns the user name.

  1. // user.js
  2. import request from './request';
  3. export function getUserName(userID) {
  4. return request('/users/' + userID).then(user => user.name);
  5. }

In the above implementation we expect the request.js module to return a promise. We chain a call to then to receive the user name.

Now imagine an implementation of request.js that goes to the network and fetches some user data:

  1. // request.js
  2. const http = require('http');
  3. export default function request(url) {
  4. return new Promise(resolve => {
  5. // This is an example of an http request, for example to fetch
  6. // user data from an API.
  7. // This module is being mocked in __mocks__/request.js
  8. http.get({path: url}, response => {
  9. let data = '';
  10. response.on('data', _data => (data += _data));
  11. response.on('end', () => resolve(data));
  12. });
  13. });
  14. }

Because we don't want to go to the network in our test, we are going to create a manual mock for our request.js module in the mocks folder (the folder is case-sensitive, MOCKS will not work). It could look something like this:

  1. // __mocks__/request.js
  2. const users = {
  3. 4: {name: 'Mark'},
  4. 5: {name: 'Paul'},
  5. };
  6. export default function request(url) {
  7. return new Promise((resolve, reject) => {
  8. const userID = parseInt(url.substr('/users/'.length), 10);
  9. process.nextTick(() =>
  10. users[userID]
  11. ? resolve(users[userID])
  12. : reject({
  13. error: 'User with ' + userID + ' not found.',
  14. }),
  15. );
  16. });
  17. }

Now let's write a test for our async functionality.

  1. // __tests__/user-test.js
  2. jest.mock('../request');
  3. import * as user from '../user';
  4. // The assertion for a promise must be returned.
  5. it('works with promises', () => {
  6. expect.assertions(1);
  7. return user.getUserName(4).then(data => expect(data).toEqual('Mark'));
  8. });

We call jest.mock('../request') to tell Jest to use our manual mock. it expects the return value to be a Promise that is going to be resolved. You can chain as many Promises as you like and call expect at any time, as long as you return a Promise at the end.

.resolves

There is a less verbose way using resolves to unwrap the value of a fulfilled promise together with any other matcher. If the promise is rejected, the assertion will fail.

  1. it('works with resolves', () => {
  2. expect.assertions(1);
  3. return expect(user.getUserName(5)).resolves.toEqual('Paul');
  4. });

async/await

Writing tests using the async/await syntax is easy. Here is how you'd write the same examples from before:

  1. // async/await can be used.
  2. it('works with async/await', async () => {
  3. expect.assertions(1);
  4. const data = await user.getUserName(4);
  5. expect(data).toEqual('Mark');
  6. });
  7. // async/await can also be used with `.resolves`.
  8. it('works with async/await and resolves', async () => {
  9. expect.assertions(1);
  10. await expect(user.getUserName(5)).resolves.toEqual('Paul');
  11. });

To enable async/await in your project, install @babel/preset-env and enable the feature in your babel.config.js file.

Error handling

Errors can be handled using the .catch method. Make sure to add expect.assertions to verify that a certain number of assertions are called. Otherwise a fulfilled promise would not fail the test:

  1. // Testing for async errors using Promise.catch.
  2. test('tests error with promises', () => {
  3. expect.assertions(1);
  4. return user.getUserName(2).catch(e =>
  5. expect(e).toEqual({
  6. error: 'User with 2 not found.',
  7. }),
  8. );
  9. });
  10. // Or using async/await.
  11. it('tests error with async/await', async () => {
  12. expect.assertions(1);
  13. try {
  14. await user.getUserName(1);
  15. } catch (e) {
  16. expect(e).toEqual({
  17. error: 'User with 1 not found.',
  18. });
  19. }
  20. });

.rejects

The.rejects helper works like the .resolves helper. If the promise is fulfilled, the test will automatically fail.

  1. // Testing for async errors using `.rejects`.
  2. it('tests error with rejects', () => {
  3. expect.assertions(1);
  4. return expect(user.getUserName(3)).rejects.toEqual({
  5. error: 'User with 3 not found.',
  6. });
  7. });
  8. // Or using async/await with `.rejects`.
  9. it('tests error with async/await and rejects', async () => {
  10. expect.assertions(1);
  11. await expect(user.getUserName(3)).rejects.toEqual({
  12. error: 'User with 3 not found.',
  13. });
  14. });

The code for this example is available at examples/async.

If you'd like to test timers, like setTimeout, take a look at the Timer mocks documentation.

原文: https://jestjs.io/docs/en/tutorial-async