// This file is part of Moonfire NVR, a security camera network video recorder. // Copyright (C) 2021 The Moonfire NVR Authors; see AUTHORS and LICENSE.txt. // SPDX-License-Identifier: GPL-v3.0-or-later WITH GPL-3.0-linking-exception import { screen } from "@testing-library/react"; import { utcToZonedTime } from "date-fns-tz"; import format from "date-fns/format"; import { rest } from "msw"; import { setupServer } from "msw/node"; import { Recording, VideoSampleEntry } from "../api"; import { renderWithCtx } from "../testutil"; import { Camera, Stream } from "../types"; import VideoList from "./VideoList"; const TEST_CAMERA: Camera = { uuid: "c7278ba0-a001-420c-911e-fff4e33f6916", shortName: "test-camera", description: "", streams: {}, }; const TEST_STREAM: Stream = { camera: TEST_CAMERA, streamType: "main", retainBytes: 0, minStartTime90k: 0, maxEndTime90k: 0, totalDuration90k: 0, totalSampleFileBytes: 0, fsBytes: 0, days: {}, }; const TEST_RANGE1: [number, number] = [ 145747836000000, // 2021-04-26T00:00:00:00000-07:00 145755612000000, // 2021-04-27T00:00:00:00000-07:00 ]; const TEST_RANGE2: [number, number] = [ 145755612000000, // 2021-04-27T00:00:00:00000-07:00 145763388000000, // 2021-04-28T00:00:00:00000-07:00 ]; const TEST_RECORDINGS1: Recording[] = [ { startId: 42, openId: 1, startTime90k: 145750542570000, // 2021-04-26T08:21:13:00000-07:00 endTime90k: 145750548150000, // 2021-04-26T08:22:15:00000-07:00 videoSampleEntryId: 4, videoSamples: 1860, sampleFileBytes: 248000, }, ]; const TEST_RECORDINGS2: Recording[] = [ { startId: 42, openId: 1, startTime90k: 145757651670000, // 2021-04-27T06:17:43:00000-07:00 endTime90k: 145757656980000, // 2021-04-27T06:18:42:00000-07:00 videoSampleEntryId: 4, videoSamples: 1860, sampleFileBytes: 248000, }, ]; const TEST_VIDEO_SAMPLE_ENTRIES: { [id: number]: VideoSampleEntry } = { 4: { width: 1920, height: 1080, }, }; function TestFormat(time90k: number) { return format( utcToZonedTime(new Date(time90k / 90), "America/Los_Angeles"), "d MMM yyyy HH:mm:ss" ); } const server = setupServer( rest.get("/api/cameras/:camera/:streamType/recordings", (req, res, ctx) => { const p = req.url.searchParams; const range90k = [ parseInt(p.get("startTime90k")!, 10), parseInt(p.get("endTime90k")!, 10), ]; if (range90k[0] === 42) { return res(ctx.status(503), ctx.text("server error")); } if (range90k[0] === TEST_RANGE1[0]) { return res( ctx.json({ recordings: TEST_RECORDINGS1, videoSampleEntries: TEST_VIDEO_SAMPLE_ENTRIES, }) ); } else { return res( ctx.delay(2000), // 2 second delay ctx.json({ recordings: TEST_RECORDINGS2, videoSampleEntries: TEST_VIDEO_SAMPLE_ENTRIES, }) ); } }) ); beforeAll(() => server.listen({ onUnhandledRequest: "error" })); afterEach(() => server.resetHandlers()); afterAll(() => server.close()); test("load", async () => { renderWithCtx( {}} formatTime={TestFormat} />
); expect(await screen.findByText(/26 Apr 2021 08:21:13/)).toBeInTheDocument(); }); // This test may be slightly flaky because it uses real timers. It looks like // msw specifically avoids using test timers: // https://github.com/mswjs/msw/pull/243 test("slow replace", async () => { const { rerender } = renderWithCtx( {}} formatTime={TestFormat} />
); expect(await screen.findByText(/26 Apr 2021 08:21:13/)).toBeInTheDocument(); rerender( {}} formatTime={TestFormat} />
); // The first results don't go away immediately. expect(screen.getByText(/26 Apr 2021 08:21:13/)).toBeInTheDocument(); // A loading indicator should show up after a second. expect(await screen.findByRole("progressbar")).toBeInTheDocument(); // Then the second query result should show up. expect(await screen.findByText(/27 Apr 2021 06:17:43/)).toBeInTheDocument(); }); test("error", async () => { renderWithCtx( {}} formatTime={TestFormat} />
); expect(await screen.findByRole("alert")).toHaveTextContent(/server error/); });