2023-02-07 21:12:39 +00:00
|
|
|
/*
|
2024-09-09 13:57:16 +00:00
|
|
|
Copyright 2024 New Vector Ltd.
|
2023-02-07 21:12:39 +00:00
|
|
|
Copyright 2023 The Matrix.org Foundation C.I.C.
|
|
|
|
|
2024-09-09 13:57:16 +00:00
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
Please see LICENSE files in the repository root for full details.
|
2023-02-07 21:12:39 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
import React from "react";
|
2024-11-20 18:09:51 +00:00
|
|
|
import { render, waitFor } from "jest-matrix-react";
|
2023-08-23 09:04:25 +00:00
|
|
|
import { EventTimeline, MatrixEvent, Room, M_TEXT } from "matrix-js-sdk/src/matrix";
|
2023-02-07 21:12:39 +00:00
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
|
|
|
|
2024-10-15 13:57:26 +00:00
|
|
|
import { IBodyProps } from "../../../../../src/components/views/messages/IBodyProps";
|
|
|
|
import { MPollEndBody } from "../../../../../src/components/views/messages/MPollEndBody";
|
|
|
|
import MatrixClientContext from "../../../../../src/contexts/MatrixClientContext";
|
|
|
|
import { RoomPermalinkCreator } from "../../../../../src/utils/permalinks/Permalinks";
|
|
|
|
import { MediaEventHelper } from "../../../../../src/utils/MediaEventHelper";
|
2023-02-07 21:12:39 +00:00
|
|
|
import {
|
|
|
|
flushPromises,
|
|
|
|
getMockClientWithEventEmitter,
|
|
|
|
makePollEndEvent,
|
|
|
|
makePollStartEvent,
|
|
|
|
mockClientMethodsEvents,
|
|
|
|
mockClientMethodsUser,
|
|
|
|
setupRoomWithPollEvents,
|
2024-10-15 13:57:26 +00:00
|
|
|
} from "../../../../test-utils";
|
2023-02-07 21:12:39 +00:00
|
|
|
|
|
|
|
describe("<MPollEndBody />", () => {
|
|
|
|
const userId = "@alice:domain.org";
|
|
|
|
const roomId = "!room:domain.org";
|
|
|
|
const mockClient = getMockClientWithEventEmitter({
|
|
|
|
...mockClientMethodsUser(userId),
|
|
|
|
...mockClientMethodsEvents(),
|
|
|
|
getRoom: jest.fn(),
|
|
|
|
relations: jest.fn(),
|
|
|
|
fetchRoomEvent: jest.fn(),
|
|
|
|
});
|
|
|
|
const pollStartEvent = makePollStartEvent("Question?", userId, undefined, { roomId });
|
|
|
|
const pollEndEvent = makePollEndEvent(pollStartEvent.getId()!, roomId, userId, 123);
|
|
|
|
|
|
|
|
const setupRoomWithEventsTimeline = async (pollEnd: MatrixEvent, pollStart?: MatrixEvent): Promise<Room> => {
|
|
|
|
if (pollStart) {
|
2023-02-12 20:19:45 +00:00
|
|
|
await setupRoomWithPollEvents([pollStart], [], [pollEnd], mockClient);
|
2023-02-07 21:12:39 +00:00
|
|
|
}
|
|
|
|
const room = mockClient.getRoom(roomId) || new Room(roomId, mockClient, userId);
|
|
|
|
|
|
|
|
// end events validate against this
|
|
|
|
jest.spyOn(room.currentState, "maySendRedactionForEvent").mockImplementation(
|
|
|
|
(_evt: MatrixEvent, id: string) => {
|
|
|
|
return id === mockClient.getSafeUserId();
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
const timelineSet = room.getUnfilteredTimelineSet();
|
|
|
|
const getTimelineForEventSpy = jest.spyOn(timelineSet, "getTimelineForEvent");
|
|
|
|
// if we have a pollStart, mock the room timeline to include it
|
|
|
|
if (pollStart) {
|
|
|
|
const eventTimeline = {
|
|
|
|
getEvents: jest.fn().mockReturnValue([pollEnd, pollStart]),
|
|
|
|
} as unknown as EventTimeline;
|
|
|
|
getTimelineForEventSpy.mockReturnValue(eventTimeline);
|
|
|
|
}
|
|
|
|
mockClient.getRoom.mockReturnValue(room);
|
|
|
|
|
|
|
|
return room;
|
|
|
|
};
|
|
|
|
|
|
|
|
const defaultProps = {
|
|
|
|
mxEvent: pollEndEvent,
|
|
|
|
highlightLink: "unused",
|
|
|
|
mediaEventHelper: {} as unknown as MediaEventHelper,
|
|
|
|
onHeightChanged: () => {},
|
|
|
|
onMessageAllowed: () => {},
|
|
|
|
permalinkCreator: {} as unknown as RoomPermalinkCreator,
|
|
|
|
ref: undefined as any,
|
|
|
|
};
|
|
|
|
|
|
|
|
const getComponent = (props: Partial<IBodyProps> = {}) =>
|
|
|
|
render(<MPollEndBody {...defaultProps} {...props} />, {
|
|
|
|
wrapper: ({ children }) => (
|
|
|
|
<MatrixClientContext.Provider value={mockClient}>{children}</MatrixClientContext.Provider>
|
|
|
|
),
|
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
mockClient.getRoom.mockReset();
|
|
|
|
mockClient.relations.mockResolvedValue({
|
|
|
|
events: [],
|
|
|
|
});
|
2024-03-25 12:21:02 +00:00
|
|
|
mockClient.fetchRoomEvent.mockResolvedValue(pollStartEvent.getEffectiveEvent());
|
2023-02-07 21:12:39 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
jest.spyOn(logger, "error").mockRestore();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when poll start event exists in current timeline", () => {
|
|
|
|
it("renders an ended poll", async () => {
|
|
|
|
await setupRoomWithEventsTimeline(pollEndEvent, pollStartEvent);
|
|
|
|
const { container } = getComponent();
|
|
|
|
|
|
|
|
// ended poll rendered
|
|
|
|
expect(container).toMatchSnapshot();
|
|
|
|
|
|
|
|
// didnt try to fetch start event while it was already in timeline
|
|
|
|
expect(mockClient.fetchRoomEvent).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("does not render a poll tile when end event is invalid", async () => {
|
|
|
|
// sender of end event does not match start event
|
|
|
|
const invalidEndEvent = makePollEndEvent(pollStartEvent.getId()!, roomId, "@mallory:domain.org", 123);
|
|
|
|
await setupRoomWithEventsTimeline(invalidEndEvent, pollStartEvent);
|
|
|
|
const { getByText } = getComponent({ mxEvent: invalidEndEvent });
|
|
|
|
|
|
|
|
// no poll tile rendered
|
|
|
|
expect(getByText("The poll has ended. Something.")).toBeTruthy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when poll start event does not exist in current timeline", () => {
|
|
|
|
it("fetches the related poll start event and displays a poll tile", async () => {
|
|
|
|
await setupRoomWithEventsTimeline(pollEndEvent);
|
2024-11-20 18:09:51 +00:00
|
|
|
const { container, getByTestId, getByRole, queryByRole } = getComponent();
|
2023-02-07 21:12:39 +00:00
|
|
|
|
|
|
|
// while fetching event, only icon is shown
|
|
|
|
expect(container).toMatchSnapshot();
|
|
|
|
|
2024-08-06 17:22:02 +00:00
|
|
|
await waitFor(() => expect(getByRole("progressbar")).toBeInTheDocument());
|
2024-11-20 18:09:51 +00:00
|
|
|
await waitFor(() => expect(queryByRole("progressbar")).not.toBeInTheDocument());
|
2023-02-07 21:12:39 +00:00
|
|
|
|
|
|
|
expect(mockClient.fetchRoomEvent).toHaveBeenCalledWith(roomId, pollStartEvent.getId());
|
|
|
|
|
|
|
|
// quick check for poll tile
|
|
|
|
expect(getByTestId("pollQuestion").innerHTML).toEqual("Question?");
|
|
|
|
expect(getByTestId("totalVotes").innerHTML).toEqual("Final result based on 0 votes");
|
|
|
|
});
|
|
|
|
|
|
|
|
it("does not render a poll tile when end event is invalid", async () => {
|
|
|
|
// sender of end event does not match start event
|
|
|
|
const invalidEndEvent = makePollEndEvent(pollStartEvent.getId()!, roomId, "@mallory:domain.org", 123);
|
|
|
|
await setupRoomWithEventsTimeline(invalidEndEvent);
|
|
|
|
const { getByText } = getComponent({ mxEvent: invalidEndEvent });
|
|
|
|
|
|
|
|
// flush the fetch event promise
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
// no poll tile rendered
|
|
|
|
expect(getByText("The poll has ended. Something.")).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("logs an error and displays the text fallback when fetching the start event fails", async () => {
|
|
|
|
await setupRoomWithEventsTimeline(pollEndEvent);
|
|
|
|
mockClient.fetchRoomEvent.mockRejectedValue({ code: 404 });
|
|
|
|
const logSpy = jest.spyOn(logger, "error").mockImplementation(() => {});
|
|
|
|
const { getByText } = getComponent();
|
|
|
|
|
|
|
|
// flush the fetch event promise
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
// poll end event fallback text used
|
|
|
|
expect(getByText("The poll has ended. Something.")).toBeTruthy();
|
|
|
|
expect(logSpy).toHaveBeenCalledWith("Failed to fetch related poll start event", { code: 404 });
|
|
|
|
});
|
|
|
|
|
|
|
|
it("logs an error and displays the extensible event text when fetching the start event fails", async () => {
|
|
|
|
await setupRoomWithEventsTimeline(pollEndEvent);
|
|
|
|
mockClient.fetchRoomEvent.mockRejectedValue({ code: 404 });
|
|
|
|
const logSpy = jest.spyOn(logger, "error").mockImplementation(() => {});
|
|
|
|
const { getByText } = getComponent();
|
|
|
|
|
|
|
|
// flush the fetch event promise
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
// poll end event fallback text used
|
|
|
|
expect(getByText("The poll has ended. Something.")).toBeTruthy();
|
|
|
|
expect(logSpy).toHaveBeenCalledWith("Failed to fetch related poll start event", { code: 404 });
|
|
|
|
});
|
|
|
|
|
|
|
|
it("displays fallback text when the poll end event does not have text", async () => {
|
|
|
|
const endWithoutText = makePollEndEvent(pollStartEvent.getId()!, roomId, userId, 123);
|
|
|
|
delete endWithoutText.getContent()[M_TEXT.name];
|
|
|
|
await setupRoomWithEventsTimeline(endWithoutText);
|
|
|
|
mockClient.fetchRoomEvent.mockRejectedValue({ code: 404 });
|
|
|
|
const { getByText } = getComponent({ mxEvent: endWithoutText });
|
|
|
|
|
|
|
|
// flush the fetch event promise
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
// default fallback text used
|
|
|
|
expect(getByText("@alice:domain.org has ended a poll")).toBeTruthy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|