From 2dc2579101730f715edc8d302d47d0decd296b89 Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 10:18:47 +0100 Subject: [PATCH 1/8] Fix SQL script syntax You cannot choose a DB when run within the tests. --- db/initdb.sql | 2 -- 1 file changed, 2 deletions(-) diff --git a/db/initdb.sql b/db/initdb.sql index a1872fda0b..be78bada95 100644 --- a/db/initdb.sql +++ b/db/initdb.sql @@ -1,5 +1,3 @@ -\c videorec; - DROP TABLE IF EXISTS videos CASCADE; CREATE TABLE videos ( From 2d34de29318cd7c7bfa7f0b84e3cbb4384225908 Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 10:20:04 +0100 Subject: [PATCH 2/8] Improve test diagnostics --- server/api.test.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/server/api.test.js b/server/api.test.js index e0de20ab9e..f8b7eca391 100644 --- a/server/api.test.js +++ b/server/api.test.js @@ -9,10 +9,10 @@ describe("/api", () => { const response = await request(app).get("/api/videos"); expect(response.statusCode).toBe(200); - expect(response.body[0].title).toBe("Never Gonna Give You Up"); - expect(response.body[0].url).toBe( - "https://www.youtube.com/watch?v=dQw4w9WgXcQ" - ); + expect(response.body).toEqual([{ + title: "Never Gonna Give You Up", + url: "https://www.youtube.com/watch?v=dQw4w9WgXcQ", + }]); }); }); @@ -31,7 +31,7 @@ describe("/api", () => { "SELECT * FROM videos WHERE id = $1", [1] ); - expect(dbResponse.rows.length).toBe(0); + expect(dbResponse.rows).toHaveLength(0); }); }); }); From b2952f059ee694690ae2e7fc524b1cfd5b0333c6 Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 10:20:12 +0100 Subject: [PATCH 3/8] Fix DELETE response status --- server/api.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/server/api.test.js b/server/api.test.js index f8b7eca391..e9fe77776d 100644 --- a/server/api.test.js +++ b/server/api.test.js @@ -21,7 +21,7 @@ describe("/api", () => { it("Returns a successful response if the id exists", async () => { const response = await request(app).delete("/api/videos/1"); - expect(response.statusCode).toBe(200); + expect(response.statusCode).toBe(204); }); it("Deletes the video from the database if the id exists", async () => { From 78dfdb4f57290fde9ebbd706460b3f02e146057c Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 10:22:38 +0100 Subject: [PATCH 4/8] Prettier code --- server/api.test.js | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/server/api.test.js b/server/api.test.js index e9fe77776d..a0f66d3144 100644 --- a/server/api.test.js +++ b/server/api.test.js @@ -9,10 +9,12 @@ describe("/api", () => { const response = await request(app).get("/api/videos"); expect(response.statusCode).toBe(200); - expect(response.body).toEqual([{ - title: "Never Gonna Give You Up", - url: "https://www.youtube.com/watch?v=dQw4w9WgXcQ", - }]); + expect(response.body).toEqual([ + { + title: "Never Gonna Give You Up", + url: "https://www.youtube.com/watch?v=dQw4w9WgXcQ", + }, + ]); }); }); From 921c99b5e0d8cc09a53787b656842e7daf953586 Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 10:28:22 +0100 Subject: [PATCH 5/8] Improve client test diagnostics --- client/src/App.test.jsx | 31 ++++++++++++++----------------- 1 file changed, 14 insertions(+), 17 deletions(-) diff --git a/client/src/App.test.jsx b/client/src/App.test.jsx index 827546e12d..788f6f1293 100644 --- a/client/src/App.test.jsx +++ b/client/src/App.test.jsx @@ -4,9 +4,10 @@ import { fireEvent, waitForElementToBeRemoved, } from "@testing-library/react"; -import { server } from "./tests/setupTests.js"; import { http, HttpResponse } from "msw"; +import { server } from "./tests/setupTests.js"; + import App from "./App.jsx"; describe("Main Page", () => { @@ -43,7 +44,7 @@ describe("Main Page", () => { ); // We have two videos, so the amount should be two - expect(videoContainers.length).toBe(2); + expect(videoContainers).toHaveLength(2); }); it("Removes the video when asked to do", async () => { @@ -67,41 +68,37 @@ describe("Main Page", () => { ); // this should now be only 1 - expect(videoContainers.length).toBe(1); + expect(videoContainers).toHaveLength(1); }); it("Adds a new video when asked to do", async () => { + const title = "New Title"; + const url = "https://www.youtube.com/watch?v=CDEYRFUTURE"; + // we set up a fake backend that allows us to send a new video. It only allows one specific title and url however server.use( http.post("/api/videos", async ({ request }) => { const data = await request.json(); - if ( - data.title != "New Title" || - data.url != "https://www.youtube.com/watch?v=CDEYRFUTURE" - ) { - return HttpResponse.json({ success: false }); + if (data.title !== title || data.url !== url) { + return HttpResponse.json({ success: false }, { status: 400 }); } - return HttpResponse.json({ - id: 3, - title: "New Title", - url: "https://www.youtube.com/watch?v=CDEYRFUTURE", - }); + return HttpResponse.json({ id: 3, title, url }); }) ); // we fill in the form fireEvent.change(screen.getByRole("textbox", { name: "Title:" }), { - target: { value: "New Title" }, + target: { value: title }, }); fireEvent.change(screen.getByRole("textbox", { name: "Url:" }), { - target: { value: "https://www.youtube.com/watch?v=CDEYRFUTURE" }, + target: { value: url }, }); // then click submit fireEvent.click(screen.getByRole("button", { name: "Submit" })); // wait for the new video to appear - await screen.findByText("New Title"); + await screen.findByText(title); // afterwards we calculate the number of videos on the page const videoContainers = screen.getAllByText( @@ -109,6 +106,6 @@ describe("Main Page", () => { ); // this should now be three - expect(videoContainers.length).toBe(3); + expect(videoContainers).toHaveLength(3); }); }); From 8e33384a5ccb313c6d270970f4b5701a2879d311 Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 10:33:55 +0100 Subject: [PATCH 6/8] Use Testing Library user events wrapper --- client/src/App.test.jsx | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/client/src/App.test.jsx b/client/src/App.test.jsx index 788f6f1293..f52a56b590 100644 --- a/client/src/App.test.jsx +++ b/client/src/App.test.jsx @@ -1,9 +1,9 @@ import { render, screen, - fireEvent, waitForElementToBeRemoved, } from "@testing-library/react"; +import userEvent from "@testing-library/user-event"; import { http, HttpResponse } from "msw"; import { server } from "./tests/setupTests.js"; @@ -11,6 +11,9 @@ import { server } from "./tests/setupTests.js"; import App from "./App.jsx"; describe("Main Page", () => { + /** @type {import("@testing-library/user-event").UserEvent} */ + let user; + beforeEach(async () => { // Here we create a fake backend that will always return two videos when calling the /api/videos endpoint server.use( @@ -35,6 +38,7 @@ describe("Main Page", () => { // Let's wait for one of the videos to appear await screen.findByText("Never Gonna Give You Up"); + user = userEvent.setup(); }); it("Renders the videos", async () => { @@ -57,7 +61,7 @@ describe("Main Page", () => { const deleteButton = screen.getAllByText("Remove video")[0]; // then we click it - fireEvent.click(deleteButton); + await user.click(deleteButton); // wait for the video to get deleted from the page await waitForElementToBeRemoved(deleteButton); @@ -87,15 +91,11 @@ describe("Main Page", () => { ); // we fill in the form - fireEvent.change(screen.getByRole("textbox", { name: "Title:" }), { - target: { value: title }, - }); - fireEvent.change(screen.getByRole("textbox", { name: "Url:" }), { - target: { value: url }, - }); + await user.type(screen.getByRole("textbox", { name: "Title:" }), title); + await user.type(screen.getByRole("textbox", { name: "Url:" }), url); // then click submit - fireEvent.click(screen.getByRole("button", { name: "Submit" })); + await user.click(screen.getByRole("button", { name: "Submit" })); // wait for the new video to appear await screen.findByText(title); From acff58a3cfac11b031bc79bbc746f77c31157f5b Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 11:08:50 +0100 Subject: [PATCH 7/8] Make fewer implementation assumptions --- client/src/App.test.jsx | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/client/src/App.test.jsx b/client/src/App.test.jsx index f52a56b590..14dd07da5a 100644 --- a/client/src/App.test.jsx +++ b/client/src/App.test.jsx @@ -1,8 +1,4 @@ -import { - render, - screen, - waitForElementToBeRemoved, -} from "@testing-library/react"; +import { render, screen, waitFor } from "@testing-library/react"; import userEvent from "@testing-library/user-event"; import { http, HttpResponse } from "msw"; @@ -58,13 +54,19 @@ describe("Main Page", () => { ); // we find the delete button on the website - const deleteButton = screen.getAllByText("Remove video")[0]; + const deleteButton = screen.getAllByRole("button", { + name: "Remove video", + })[0]; // then we click it await user.click(deleteButton); // wait for the video to get deleted from the page - await waitForElementToBeRemoved(deleteButton); + await waitFor(() => + expect( + screen.getAllByRole("button", { name: "Remove video" }) + ).toHaveLength(1) + ); // we calculate the number of videos after the call const videoContainers = screen.getAllByText( From 34460ff4aa8cc8a1592f7137c3d6878b7ad8bb33 Mon Sep 17 00:00:00 2001 From: Jonathan Sharpe Date: Sun, 2 Jun 2024 11:09:05 +0100 Subject: [PATCH 8/8] Match backend tests --- client/src/App.test.jsx | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/client/src/App.test.jsx b/client/src/App.test.jsx index 14dd07da5a..57f54c9670 100644 --- a/client/src/App.test.jsx +++ b/client/src/App.test.jsx @@ -50,7 +50,10 @@ describe("Main Page", () => { it("Removes the video when asked to do", async () => { // we create another fake backend that listens on the delete call, and returns success server.use( - http.delete("/api/videos/1", () => HttpResponse.json({ success: true })) + http.delete( + "/api/videos/1", + () => new HttpResponse(null, { status: 204 }) + ) ); // we find the delete button on the website