在這篇部落格文章中,我們將探討使用 Jest 和 SuperTest 來測試 GraphQL API 所涉及的挑戰和解決方案。這個旅程始於需要在 Jest 測試中模擬標頭,特別是基於令牌的身份驗證。
在為 Woovi 挑戰開發 Todo 後端 GraphQL 專案時,我遇到了一個重大障礙。我需要測試 GraphQL API 的身份驗證,該驗證依賴於 HTTP 標頭中傳遞的 JSON Web 令牌 (JWT)。最初,我努力尋找一種直接的方法來在 Jest 中模擬這些標頭。標準 Jest 設定還不夠,因為它沒有像真實伺服器那樣直接處理 HTTP 請求和回應。
經過多次嘗試和錯誤,我遇到了 SuperTest,一個專為 HTTP 斷言設計的函式庫。 SuperTest 可讓您像測試真實客戶端一樣測試 HTTP 伺服器,從而擴展了 Jest 的功能。此功能使得模擬標頭成為可能,包括我的 API 驗證所需的授權令牌。
在進行測試之前,讓我們先設定環境。
npm install --save-dev jest supertest faker
module.exports = { preset: 'ts-jest', testEnvironment: 'node', };
SuperTest 成為了這種情況下的遊戲規則改變者。以下是我如何使用它來測試 API 的 CRUD 操作和身份驗證。
import { connect, disconnectDatabase } from './mongooseConnection'; import supertest from 'supertest'; import app from './../index'; beforeAll(async () => { await connect(); }); afterAll(async () => { await disconnectDatabase(); });
import { faker } from '@faker-js/faker'; import { graphql } from 'graphql'; import { schema } from '../schema'; async function authUserTest() { const userTest = { name: faker.name.firstName(), email: faker.internet.email(), password: faker.internet.password(), }; const source = ` mutation { register(name: "${userTest.name}", email: "${userTest.email}", password: "${userTest.password}") { token user { name email } } } `; const result = await graphql({ schema, source }); const data = result.data?.register; return data.token; }
測試任務 CRUD 操作
it('should create a new task', async () => { const todo = { task: faker.lorem.words(), status: faker.helpers.arrayElement(['pending', 'complete', 'in progress']), }; const query = ` mutation { todo(task: "${todo.task}", status: "${todo.status}") { task status } } `; const { body } = await supertest(app) .post('/graphql') .send({ query }) .set('Accept', 'application/json') .set('Authorization', `Bearer ${await authUserTest()}`); expect(body.data.todo).toMatchObject(todo); });
擷取所有任務
it('should retrieve all tasks', async () => { const query = ` query { todos { _id task status } } `; const { body } = await supertest(app) .post('/graphql') .send({ query }) .set('Accept', 'application/json') .set('Authorization', `Bearer ${await authUserTest()}`); expect(body.data.todos).toBeInstanceOf(Array); });
更新任務
it('should update a task', async () => { const todos = await Todo.find(); const randomTodo = todos[Math.floor(Math.random() * todos.length)]; const updatedTask = faker.lorem.words(); const updatedStatus = faker.helpers.arrayElement(['pending', 'complete', 'in progress']); const query = ` mutation { updateTodo(_id: "${randomTodo._id}", task: "${updatedTask}", status: "${updatedStatus}") { task status } } `; const { body } = await supertest(app) .post('/graphql') .send({ query }) .set('Accept', 'application/json') .set('Authorization', `Bearer ${await authUserTest()}`); expect(body.data.updateTodo.task).toBe(updatedTask); expect(body.data.updateTodo.status).toBe(updatedStatus); });
刪除任務
it('should delete a task', async () => { const todos = await Todo.find(); const randomTodo = todos[Math.floor(Math.random() * todos.length)]; const query = ` mutation { deleteTodo(_id: "${randomTodo._id}") { _id } } `; const { body } = await supertest(app) .post('/graphql') .send({ query }) .set('Accept', 'application/json') .set('Authorization', `Bearer ${await authUserTest()}`); expect(body.data.deleteTodo._id).toBe(randomTodo._id); });
使用 Jest 執行測試:
npm test
此指令將執行所有測試文件,並提供詳細的結果報告。
在 Jest 中模擬 header 的困難導致了 SuperTest 的發現,它大大簡化了過程。透過利用 SuperTest 和 Jest,我能夠有效地測試 GraphQL API 的身份驗證和 CRUD 操作,確保應用程式的安全性和功能。分享這個學習過程突顯了公共學習和社區驅動的問題解決的力量。
以上是使用 Jest 和 SuperTest 測試 GraphQL 應用程式的詳細內容。更多資訊請關注PHP中文網其他相關文章!