Easy API Test lets you write tests to check that you're API is behaving the way you expect it to. The tests are run in the order you decide, so you easily can test an entire user flow, and it now supports WebSocket!
The libary's API is meant to be simple. No CLI and no need to do extra configuration to get your Babel or Typescript tests to work.
Uses the great SuperTest package for the request helper functions.
You can find the full API reference here.
-
Using NPM:
npm install -D easy-api-test
-
Using Yarn:
yarn add -D easy-api-test
// test.js
const { use, test, get, localStorage } = require('easy-api-test')
// Tell Easy API Test which URL we'll be testing
use('http://localhost:3000/')
// Define a first test
test('Create a user', async () => {
const response = await post('/users')
.send({ username: 'Nay' })
.expect(200)
// Save the user for later use
localStorage.setItem('user', response.body)
})
// Define a second test
test('Get a user', async () => {
// Tests run in the defined order, so localStorage will always contain the user.
const user = localStorage.getItem('user')
await get(`/users/${user.id}`)
.expect(200, user)
})
// Run the tests
run()
To run the tests that you wrote in test.js
, you simply need to execute the script:
-
With JavaScript (node):
node test.js
-
With ES6+ or Flow (babel-node):
babel-node test.js
-
With Typescript (ts-node):
ts-node test.ts
The code of the example is available here.
-
Let's say we are building an express app that exposes this API:
- / {*} -> Say 'hello' - /recipes {GET} -> Get all the recipes - /recipes/:id {GET} -> Get a recipe by ID - /recipes {POST} -> Create a recipe
First, let's create a new folder:
mkdir my-recipe-api
Move to the project's directory:
cd my-recipe-api
Initialize a new NPM project:
npm init -y
-
We are going to use express, and body-parser to parse JSON. Let's install them:
npm install --save express body-parser
And for our tests, easy-api-test is all that we will need, let's install it as a dev dependency:
npm install --save-dev easy-api-test
-
To keep things clean, we are going to create an
src/
folder that will contain our code, and atest/
folder that will contain our test files.Let's start by writing our app in the
src/
folder, in a file that we will nameapp.js
.// src/app.js const http = require('http') // Import express and body-parser const express = require('express') const bodyParser = require('body-parser') // Create a new express app const app = express() // This is were we will store the recipes const recipes = [] // Add the bodyParser's json middleware to our app app.use(bodyParser.json()) // We return 'hello' at the root, regardless of the method app.all('/', (req, res) => { res.end('hello') }) // We return all the recipes at /recipes app.get('/recipes', (req, res) => { res.json(recipes) }) // We return a recipe by id app.get('/recipes/:id', (req, res) => { const id = req.params.id // Check if the recipe exists if (recipes[id]) { return res.json(recipes[id]) } // Send a 404 'Not found' error if the id does not exist. res.statusCode = 404 res.end() }) // We create a new recipe app.post('/recipes', (req, res) => { // Check if the fields are correct if (req.body.name && req.body.content) { // Create the recipe const recipe = { id: recipes.length, name: req.body.name, content: req.body.content, } recipes.push(recipe) // And return it return res.json(recipe) } // Send a 400 'Bad request' error if the fields were not correct res.statusCode = 400 res.end() }) // Wrap the app into a Server instance and export it module.exports = http.createServer(app)
We can create an
src/index.js
file to run our app:// src/index.js const app = require('./app') // Start the app app.listen()
-
Our tests will be written in the
test/
folder, let's start by creating atest/index.js
file:// test/index.js // Import what we'll need and our server const { use, start, end, run } = require('easy-api-test') const app = require('../src/app') // We can tell Easy API Test which URL to test, // here we will use the port 3000 use('http://localhost:3000/') // Any function passed to `start()` will run before the tests start(() => { // Return a promise that resolves once the server started return new Promise(resolve => { app.listen(3000, resolve) }) }) // Any function passed to `stop()` will run after the tests, even // if a test has failed. end(() => { app.close() }) /** * We will import our tests here */ // Run the tests. run()
What this file does, for now, is:
- Telling Easy API Test that we will be testing
http://localhost:3000/
. - Start the server before the tests begin
- Close the server after the tests ran, or after a test failed.
Now, we will write our tests! Let's start by testing the root
/
path of our API. Remember? It should always return 'hello', let's write a test intest/01-root.js
://test/01-root.js const { suite, test, get, post } = require('easy-api-test') // Suites are optional, but they let us group tests together, which // makes visual feedback clearer suite('/ Root', () => { // Tests require a name, and a function to run test('Say hello', async () => { await get('/').expect(200, 'hello') // Expect status 200 and content 'hello' }) test('Say hello even with the POST method', async () => { await post('/').expect(200, 'hello') }) })
Tests will run in the same order as you declared them.
It is a good idea to start your test file names with a number so your file explorer or text editor or brain can list them in ascending order.
Here, we used
get()
andpost()
. These are helper methods built on top of SuperTest, hence SuperTest's API is available for you to use, along with some extra methods.Anything that throws will work in the tests, so feel free to use any library you wish!
So, we created our tests for
/
, but we now need to add them to our test pipeline intest/index.js
:// test/index.js const { use, start, end, run } = require('easy-api-test') const app = require('../src/app') use('http://localhost:3000/') start(() => { return new Promise(resolve => { app.listen(3000, resolve) }) }) end(() => { app.close() }) /** * We import our tests here */ require('./01-root') // <====== here run()
Next, we want to test our
/recipes
endpoint. We don't have any recipe yet, so let's test if our API creates recipes as expected in a new filetest/create-recipe.js
:// test/02-create-recipe.js const assert = require('assert') const { suite, test, post, localStorage } = require('easy-api-test') suite('/recipes Create new recipes', () => { // Let's first see if our API rejects invalid recipes test('Reject an invalid recipe', async () => { // We expect a 400 error in case of incorrect fields await post('/recipes').send({ ayaya: 'clap' }).expect(400) }) // Now let's test creating a recipe test('Create a valid recipe', async () => { const recipe = { name: 'tomatoes', content: 'Put tomatoes in a plate.', } const response = await post('/recipes').send(recipe) // We can use `assert` to test anything assert.strictEqual(response.status, 200) assert.strictEqual(response.body.name, recipe.name) // localStorage is a helper object from Easy API Test, to store data // and retrieve it easily in later tests. localStorage.setItem('recipe', response.body) }) })
It might sound scary to have stateful tests where one test requires the result of another test, but don't worry, this actually is intended! We are not testing units of our application, we are testing an actual user flow through the API. If one test crashes, the test suites will end, so you will not have a test missing some external data.
Let's add this new suite to our pipeline in
test/index.js
:// test/index.js const { use, start, end, run } = require('easy-api-test') const app = require('../src/app') use('http://localhost:3000/') start(() => { return new Promise(resolve => { app.listen(3000, resolve) }) }) end(() => { app.close() }) /** * We import our tests here */ require('./01-root') require('./02-create-recipe') // <====== here run()
And let's finish by writing tests for the
/recipes
GET endpoint, and/recipes/:id
endpoint in a file nammedtest/get-recipe.js
:// test/03-get-recipe.js const assert = require('assert') const { suite, test, get, localStorage } = require('easy-api-test') suite('/recipes Get recipes', () => { // This will run after we have created our recipe, so if the endpoint works, // it should return an array with at least 1 element. test('Get all the recipes', async () => { const response = await get('/recipes') assert(response.body.length > 0) }) test('Get a recipe by id', async () => { // Our recipe is in the localStorage, so we can grab its ID const recipe = localStorage.getItem('recipe') await get(`/recipes/${recipe.id}`).expect(200, recipe) }) // Check if an invalid ID returns 404 test('Get 404 when the recipe does not exist', async () => { await get('/recipes/ayaya').expect(404) }) })
And again, add this to our pipeline:
// test/index.js const { use, start, end, run } = require('easy-api-test') const app = require('../src/app') use('http://localhost:3000/') start(() => { return new Promise(resolve => { app.listen(3000, resolve) }) }) end(() => { app.close() }) /** * We import our tests here */ require('./01-root') require('./02-create-recipe') require('./03-get-recipe') // <====== here run()
That's it! Now let's see if our app works as expected by running the tests:
node test/index.js
and the result:
/ Root: - Say hello: PASS (29ms) - Say hello even with the POST method: PASS (4ms) /recipes Create new recipes: - Reject an invalid recipe: PASS (15ms) - Create a valid recipe: PASS (6ms) /recipes Get recipes: - Get all the recipes: PASS (3ms) - Get a recipe by id: PASS (4ms) - Get 404 when the recipe does not exist: PASS (2ms)
- Telling Easy API Test that we will be testing
It is common to have data go through both an HTTP API and WebSocket events, but how to test, for example, that a certain HTTP request from a user will trigger a WebSocket event to another user? This is what we are going to see now.
All the code for this example is available here
-
We'll write a very minimal example here, just to show you how to use the WebSocket testing utility. Our application will consist of some kind of echo server. It will send back every message it receives, prepended with 'You said: '.
For example, if we send 'Hello Server', we would receive 'You said: Hello Server'.
-
Let's first create a new directory:
mkdir websocket-example
-
Initialize a new NPM project:
npm init -y
-
And install what we will need:
npm install ws easy-api-test
We will be using ws for the WebSocket server. This is also the library used by Easy API Test for the WebSocket testing utility.
Let's create an
src/
folder and write some code in it:// src/index.js const { createServer } = require('http') const ws = require('ws') // Create an HTTP server const server = createServer() // Create a websocket server on port 3000 const wss = new ws.Server({ server }) // Listen to connection events wss.on('connection', ws => { // Listen to socket message events ws.on('message', data => { // Send 'You said: ' followed by the message. ws.send(`You said: ${data.toString()}`) }) }) server.listen(3000) module.exports = server
This code imports the createServer function from the NodeJS
http
module, as well as thews
package that will let us create a WebSocket server.We then create an HTTP server, and a WebSocket server attached to that HTTP server, and tell our WebSocket server to listen to socket messages and send them back prepending 'You said: ' to each of them.
The last two instructions make the server listen on port
3000
, and export the server so that we canrequire()
it in our tests and close it once our tests are done. If we don't close the server, the NodeJS process will not exit on its own. T-TNow let's write some tests, we can create a
test/
directory at the root of our project, and write our tests. We'll use a single file because there is not much to test:// test/index.js const { test, WebSocket, end, run } = require('easy-api-test') const assert = require('assert') // Import the server const server = require('../src') test('Connect to WebSocket server', async () => { // Connect to the websocket server const ws = await WebSocket('ws://localhost:3000') assert(ws.connected === true) }) test('Echo the messages sent to the server', async () => { const ws = await WebSocket('ws://localhost:3000') // Send 'Hello' ws.send('Hello') // Get the next message from the server const message = await ws.message // Check if it is what we expected assert(message === 'You said: Hello') }) end(() => { // Close the server after the tests end server.close() }) // Run the tests run()
So, to create a WebSocket connection, simply import WebSocket from
easy-api-test
, and call the function. It returns a promise that resolves into a WebSocketClient object.The WebSocketClient object lets you easily get the incoming WebSocket messages, send messages and even automatically parse and serialize the messages. Every opened WebSocketClient is closed automatically upon tests ending or failing, so no need to manually close those connections.
-
We can now run our tests:
node test/index.js
-
You should be getting this output:
- Connect to WebSocket server: PASS (23ms) - Echo the messages sent to the server: PASS (5ms)
-
It is working! We can now start our app and enjoy that high quality echo server:
node src/index.js
Of course, you can test API calls using the
get
,post
, etc... helpers and WebSocket messages using theWebSocket
utility together in the same tests. Let's say you want to test a realtime chat that uses HTTP to send messages, and WebSocket to receive messages, you could send a message using:post('path/to/messages') .send({ message: 'Hello uwu' })
and listen to the WebSocket's next message using:
const message = await ws.message
just like we did just before. Make sure to use
await ws.message
afterpost()
otherwise theawait
keyword will block execution. -
The Code of Conduct is available here.
If you wish to help to improve Easy API Test by adding new features, performance improvements or documentation corrections, you can do so by forking the repository, applying your changes and sending a pull request.
If you enconter any issue regarding the repository, or wish to suggest new features, you can fill an issue. Make sure to check that your issue doesn't exist yet.