In this tutorial, we explain why and how to use Playwright for API testing, as well as provide an in-depth tutorial that will walk you through a live example of API testing.
Playwright is a great tool, build by Microsoft, primarily for end-to-end UI testing. But, Playwright also has excellent API capabilities you can leverage to test server functionality, setup and teardown context for your tests, or verify your backend state.
When to Use Playwright for API testing?
There are a few use-cases for Playwright for API testing:
1. Assert Correct Backend State
As part of your end to end test, you might want to query the backend directly, to ensure changes are reflected in your DB. It’s usually best, to verify changes through the UI. That way you make sure that the entire experience works. But, when the data is not presented, for example when a certain action should trigger an email sent, you should use Playwright API tests
2. Prepare the app state for testing (e.g. login)
Before every test case, you’ll need to prepare the app state. The more obvious part is logging in, but for example, if you have a test that edits an entity in your app, you might want to create that entity before the test. That’ll help you keep the data consistent.
Either way, preparing the state through the API rather than the UI is preferable for two reasons:
- Speed: calling to the API directly is faster
- Isolated: When testing a functionality, you want to isolate the test as much as possible. Doing the setup through an API, reduces the chances that bugs in other part of your app that are unrelated to the current test case, will affect it.
3. Backend API Testing
Even outside the scope of UI testing, Playwright can be a great tool to simply test your API gateway. Using Playwight allows you to consolidate setup, config and pipelines between your UI and Backend tests.
Tutorial Overview
This tutorial will demonstrate how to use Playwright for API testing on Conduit -an open-source Medium clone.
We will create a simple test that edits an article title. For that we will do a few things:
- Setup global authentication using Conduit API, store the auth token in
.auth
file to be used by all tests - Create a spec file for all tests related to “Articles Edit”.
- In the beforeEach hook, make an API request to create an article. Remember, we don’t want to test the “Create article” feature, only create one for the other tests.
- In the test itself, edit the article using the UI. Then validate that the change happened by querying the backend
- In the afterEach hook, delete the article.
Authenticate using Playwright APIRequest class.
- Go to your project directory and run the following command. Answer the questions using the default values to get your project set up.
- Delete all files inside the
tests
directory and create a file namedauth.setup.ts
with the following code:
Let’s go over what the code above does:
- Makes a POST with the user name and password in the request body to authenticate through the API.
- Validates that we received a 200 status code in the api response.
- Saves the token from the response body to
.auth
so we can read it later in the spec files. Conduit authentication works by saving the token to the browser local storage, under the value “jwtToken”, so that’s why we are not just saving the token as is, but constructing it into a storageState object.
Next open playwright.config.ts
and replace the projects key with the following configuration options:
You can read more about this setup here, but in short we define a setup project that runs auth.setup.ts
file. Then, we config our test project have a dependency on “setup” and load .auth
to the browser context so we are already authenticated witnin our test file.
Finally run:
And make sure our test passes
Use Playwright to setup test context
Under /tests
folder create a file named ArticleEdit.spec.ts
. Add the following code
A few things are going on in this code snippet:
- Get the JWT from local storage, so we can use it when making API requests .
- Create a new APIRequestContext with a base URL and an Authorization header. Going forward, every time we make a request, the auth token will already be included.
- Make a POST request to create an article and save the response into
articleObject
- Go to the editor URL so we are ready to begin testing.
All of this is happening in the beforeEach hook, meaning that this setup code will run before every test.
Go head and run the test. After you run the test open your browser and go to https://demo.realworld.io/#/@pw-api-tutorial login with username pw-api-tutorial@example.com
and password 1234
and verify that the article was created.
Use Playwright to teardown a test
Before we write the test itself, let’s just delete the article in the afterEach hook. Add the code snippet below to your test:
This code add an afterEach hook that will run after every test in the ArticleEdit.spec.ts
file and will delete the article that was created in the beforeEach
hook.
Use Playwright to verify DB state after an action
Now that we have hooks that create and delete an article, we are ready to write the test itself. Add the following code to ArticleEdit.spec.ts
The code above is pretty straight forward. At the first section, we add a tag, then, assert that the tag elements are visible, and finally we use the API to get the article object and assert the tags are there.
One last time, run npx playwright test
. Let’s review the html report
Final thoughts
In this tutorial, we’ve demonstrated how you can use Playwright API testing feature to setup seed data for your tests, keep your data consistent, verify the backend state and finally clean up the data. Using the API results in a more isolated, faster and overall robust tests.
If you’d like to take this tutorial even further, there are a few paths to improve our setup:
Use fixtures to create a global APIRequestContext
In the test we wrote, we create an APIRequestContext that automatically has a baseURL and adds the JWT to the headers. But want if we want to create a global APIRequestContext that will be used across test files?
One way is to expend the request fixture. We will not go into that in details in this article, but you should go over the official Playwright docs.
Develop a data factory to create articles and seed data more consistently
There might be many other tests that need to create and delete an article via the API as part of the test. A more robust way to approach these kind of tasks, is to create a “Data Factory”. Essentially a set of functions that create fake data objects and use the API to generate them. So in a test, when you need to create an article, all you need to do is write something like DataFactor.Articles.createArticle()
. You can use the Faker library to easily create randomized fake data.
Gal Vered
Gal Vered is a Co-Founder at Checksum where they use AI to generate end-to-end Cypress and Playwright tests, so that dev teams know that their product is thoroughly tested and shipped bug free, without the need to manually write or maintain tests.
In his role, Gal helped many teams build their testing infrastructure, solve typical (and not so typical) testing challenges and deploy AI to move fast and ship high quality software.