2024-08-21 16:50:00 +08:00
|
|
|
/*
|
2024-09-09 21:57:16 +08:00
|
|
|
* Copyright 2024 New Vector Ltd.
|
2024-08-21 16:50:00 +08:00
|
|
|
* Copyright 2024 The Matrix.org Foundation C.I.C.
|
|
|
|
*
|
2024-09-09 21:57:16 +08:00
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
* Please see LICENSE files in the repository root for full details.
|
2024-08-21 16:50:00 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
import { EventTimeline, EventType, IEvent, MatrixClient, MatrixEvent, Room } from "matrix-js-sdk/src/matrix";
|
|
|
|
import { mocked } from "jest-mock";
|
|
|
|
|
2024-10-15 21:57:26 +08:00
|
|
|
import { createTestClient } from "../../test-utils";
|
|
|
|
import PinningUtils from "../../../src/utils/PinningUtils";
|
|
|
|
import SettingsStore from "../../../src/settings/SettingsStore";
|
|
|
|
import { isContentActionable } from "../../../src/utils/EventUtils";
|
|
|
|
import { ReadPinsEventId } from "../../../src/components/views/right_panel/types";
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-10-15 21:57:26 +08:00
|
|
|
jest.mock("../../../src/utils/EventUtils", () => {
|
2024-08-21 16:50:00 +08:00
|
|
|
return {
|
|
|
|
isContentActionable: jest.fn(),
|
|
|
|
canPinEvent: jest.fn(),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("PinningUtils", () => {
|
|
|
|
const roomId = "!room:example.org";
|
|
|
|
const userId = "@alice:example.org";
|
|
|
|
|
|
|
|
const mockedIsContentActionable = mocked(isContentActionable);
|
|
|
|
|
|
|
|
let matrixClient: MatrixClient;
|
|
|
|
let room: Room;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a pinned event with the given content.
|
|
|
|
* @param content
|
|
|
|
*/
|
|
|
|
function makePinEvent(content?: Partial<IEvent>) {
|
|
|
|
return new MatrixEvent({
|
|
|
|
type: EventType.RoomMessage,
|
|
|
|
sender: userId,
|
|
|
|
content: {
|
|
|
|
body: "First pinned message",
|
|
|
|
msgtype: "m.text",
|
|
|
|
},
|
|
|
|
room_id: roomId,
|
|
|
|
origin_server_ts: 0,
|
|
|
|
event_id: "$eventId",
|
|
|
|
...content,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
// Enable feature pinning
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockReturnValue(true);
|
|
|
|
mockedIsContentActionable.mockImplementation(() => true);
|
|
|
|
|
|
|
|
matrixClient = createTestClient();
|
|
|
|
room = new Room(roomId, matrixClient, userId);
|
|
|
|
matrixClient.getRoom = jest.fn().mockReturnValue(room);
|
|
|
|
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"mayClientSendStateEvent",
|
|
|
|
).mockReturnValue(true);
|
|
|
|
});
|
|
|
|
|
2024-08-29 22:26:10 +08:00
|
|
|
describe("isUnpinnable", () => {
|
2024-08-21 16:50:00 +08:00
|
|
|
test.each(PinningUtils.PINNABLE_EVENT_TYPES)("should return true for pinnable event types", (eventType) => {
|
|
|
|
const event = makePinEvent({ type: eventType });
|
2024-08-29 22:26:10 +08:00
|
|
|
expect(PinningUtils.isUnpinnable(event)).toBe(true);
|
2024-08-21 16:50:00 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
test("should return false for a non pinnable event type", () => {
|
|
|
|
const event = makePinEvent({ type: EventType.RoomCreate });
|
2024-08-29 22:26:10 +08:00
|
|
|
expect(PinningUtils.isUnpinnable(event)).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should return true for a redacted event", () => {
|
|
|
|
const event = makePinEvent({ unsigned: { redacted_because: "because" as unknown as IEvent } });
|
|
|
|
expect(PinningUtils.isUnpinnable(event)).toBe(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("isPinnable", () => {
|
|
|
|
test.each(PinningUtils.PINNABLE_EVENT_TYPES)("should return true for pinnable event types", (eventType) => {
|
|
|
|
const event = makePinEvent({ type: eventType });
|
|
|
|
expect(PinningUtils.isPinnable(event)).toBe(true);
|
2024-08-21 16:50:00 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
test("should return false for a redacted event", () => {
|
|
|
|
const event = makePinEvent({ unsigned: { redacted_because: "because" as unknown as IEvent } });
|
|
|
|
expect(PinningUtils.isPinnable(event)).toBe(false);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("isPinned", () => {
|
|
|
|
test("should return false if no room", () => {
|
|
|
|
matrixClient.getRoom = jest.fn().mockReturnValue(undefined);
|
|
|
|
const event = makePinEvent();
|
|
|
|
|
|
|
|
expect(PinningUtils.isPinned(matrixClient, event)).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should return false if no pinned event", () => {
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"getStateEvents",
|
|
|
|
).mockReturnValue(null);
|
|
|
|
|
|
|
|
const event = makePinEvent();
|
|
|
|
expect(PinningUtils.isPinned(matrixClient, event)).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should return false if pinned events do not contain the event id", () => {
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"getStateEvents",
|
|
|
|
).mockReturnValue({
|
|
|
|
// @ts-ignore
|
|
|
|
getContent: () => ({ pinned: ["$otherEventId"] }),
|
|
|
|
});
|
|
|
|
|
|
|
|
const event = makePinEvent();
|
|
|
|
expect(PinningUtils.isPinned(matrixClient, event)).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should return true if pinned events contains the event id", () => {
|
|
|
|
const event = makePinEvent();
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"getStateEvents",
|
|
|
|
).mockReturnValue({
|
|
|
|
// @ts-ignore
|
|
|
|
getContent: () => ({ pinned: [event.getId()] }),
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(PinningUtils.isPinned(matrixClient, event)).toBe(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
describe("canPin & canUnpin", () => {
|
|
|
|
describe("canPin", () => {
|
|
|
|
test("should return false if event is not actionable", () => {
|
|
|
|
mockedIsContentActionable.mockImplementation(() => false);
|
|
|
|
const event = makePinEvent();
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
expect(PinningUtils.canPin(matrixClient, event)).toBe(false);
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
test("should return false if no room", () => {
|
|
|
|
matrixClient.getRoom = jest.fn().mockReturnValue(undefined);
|
|
|
|
const event = makePinEvent();
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
expect(PinningUtils.canPin(matrixClient, event)).toBe(false);
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
test("should return false if client cannot send state event", () => {
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"mayClientSendStateEvent",
|
|
|
|
).mockReturnValue(false);
|
|
|
|
const event = makePinEvent();
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
expect(PinningUtils.canPin(matrixClient, event)).toBe(false);
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
test("should return false if event is not pinnable", () => {
|
|
|
|
const event = makePinEvent({ type: EventType.RoomCreate });
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
expect(PinningUtils.canPin(matrixClient, event)).toBe(false);
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
test("should return true if all conditions are met", () => {
|
|
|
|
const event = makePinEvent();
|
|
|
|
|
|
|
|
expect(PinningUtils.canPin(matrixClient, event)).toBe(true);
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
});
|
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
describe("canUnpin", () => {
|
|
|
|
test("should return false if event is not unpinnable", () => {
|
|
|
|
const event = makePinEvent({ type: EventType.RoomCreate });
|
2024-08-21 16:50:00 +08:00
|
|
|
|
2024-09-05 22:37:24 +08:00
|
|
|
expect(PinningUtils.canUnpin(matrixClient, event)).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should return true if all conditions are met", () => {
|
|
|
|
const event = makePinEvent();
|
|
|
|
|
|
|
|
expect(PinningUtils.canUnpin(matrixClient, event)).toBe(true);
|
|
|
|
});
|
2024-10-02 16:23:22 +08:00
|
|
|
|
|
|
|
test("should return true if the event is redacted", () => {
|
|
|
|
const event = makePinEvent({ unsigned: { redacted_because: "because" as unknown as IEvent } });
|
|
|
|
|
|
|
|
expect(PinningUtils.canUnpin(matrixClient, event)).toBe(true);
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("pinOrUnpinEvent", () => {
|
|
|
|
test("should do nothing if no room", async () => {
|
|
|
|
matrixClient.getRoom = jest.fn().mockReturnValue(undefined);
|
|
|
|
const event = makePinEvent();
|
|
|
|
|
|
|
|
await PinningUtils.pinOrUnpinEvent(matrixClient, event);
|
|
|
|
expect(matrixClient.sendStateEvent).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should do nothing if no event id", async () => {
|
|
|
|
const event = makePinEvent({ event_id: undefined });
|
|
|
|
|
|
|
|
await PinningUtils.pinOrUnpinEvent(matrixClient, event);
|
|
|
|
expect(matrixClient.sendStateEvent).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should pin the event if not pinned", async () => {
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"getStateEvents",
|
|
|
|
).mockReturnValue({
|
|
|
|
// @ts-ignore
|
|
|
|
getContent: () => ({ pinned: ["$otherEventId"] }),
|
|
|
|
});
|
|
|
|
|
|
|
|
jest.spyOn(room, "getAccountData").mockReturnValue({
|
|
|
|
getContent: jest.fn().mockReturnValue({
|
|
|
|
event_ids: ["$otherEventId"],
|
|
|
|
}),
|
|
|
|
} as unknown as MatrixEvent);
|
|
|
|
|
|
|
|
const event = makePinEvent();
|
|
|
|
await PinningUtils.pinOrUnpinEvent(matrixClient, event);
|
|
|
|
|
|
|
|
expect(matrixClient.setRoomAccountData).toHaveBeenCalledWith(roomId, ReadPinsEventId, {
|
|
|
|
event_ids: ["$otherEventId", event.getId()],
|
|
|
|
});
|
|
|
|
expect(matrixClient.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
roomId,
|
|
|
|
EventType.RoomPinnedEvents,
|
|
|
|
{ pinned: ["$otherEventId", event.getId()] },
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should unpin the event if already pinned", async () => {
|
|
|
|
const event = makePinEvent();
|
|
|
|
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"getStateEvents",
|
|
|
|
).mockReturnValue({
|
|
|
|
// @ts-ignore
|
|
|
|
getContent: () => ({ pinned: [event.getId(), "$otherEventId"] }),
|
|
|
|
});
|
|
|
|
|
|
|
|
await PinningUtils.pinOrUnpinEvent(matrixClient, event);
|
|
|
|
expect(matrixClient.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
roomId,
|
|
|
|
EventType.RoomPinnedEvents,
|
|
|
|
{ pinned: ["$otherEventId"] },
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2024-09-05 22:37:24 +08:00
|
|
|
|
|
|
|
describe("userHasPinOrUnpinPermission", () => {
|
|
|
|
test("should return true if user can pin or unpin", () => {
|
|
|
|
expect(PinningUtils.userHasPinOrUnpinPermission(matrixClient, room)).toBe(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("should return false if client cannot send state event", () => {
|
|
|
|
jest.spyOn(
|
|
|
|
matrixClient.getRoom(roomId)!.getLiveTimeline().getState(EventTimeline.FORWARDS)!,
|
|
|
|
"mayClientSendStateEvent",
|
|
|
|
).mockReturnValue(false);
|
|
|
|
|
|
|
|
expect(PinningUtils.userHasPinOrUnpinPermission(matrixClient, room)).toBe(false);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("unpinAllEvents", () => {
|
|
|
|
it("should unpin all events in the given room", async () => {
|
|
|
|
await PinningUtils.unpinAllEvents(matrixClient, roomId);
|
|
|
|
|
|
|
|
expect(matrixClient.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
roomId,
|
|
|
|
EventType.RoomPinnedEvents,
|
|
|
|
{ pinned: [] },
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2024-08-21 16:50:00 +08:00
|
|
|
});
|