2022-06-07 20:20:32 +00:00
|
|
|
/*
|
2024-09-09 13:57:16 +00:00
|
|
|
Copyright 2024 New Vector Ltd.
|
2022-06-07 20:20:32 +00:00
|
|
|
Copyright 2022 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.
|
2022-06-07 20:20:32 +00:00
|
|
|
*/
|
|
|
|
|
2024-03-18 14:40:52 +00:00
|
|
|
import { MatrixClient, Room, RoomMember } from "matrix-js-sdk/src/matrix";
|
|
|
|
import { KnownMembership } from "matrix-js-sdk/src/types";
|
2022-08-16 13:20:26 +00:00
|
|
|
import { mocked } from "jest-mock";
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2024-10-15 13:57:26 +00:00
|
|
|
import { Command, Commands, getCommand } from "../../src/SlashCommands";
|
|
|
|
import { createTestClient } from "../test-utils";
|
|
|
|
import { LocalRoom, LOCAL_ROOM_ID_PREFIX } from "../../src/models/LocalRoom";
|
|
|
|
import SettingsStore from "../../src/settings/SettingsStore";
|
|
|
|
import LegacyCallHandler from "../../src/LegacyCallHandler";
|
|
|
|
import { SdkContextClass } from "../../src/contexts/SDKContext";
|
|
|
|
import Modal from "../../src/Modal";
|
|
|
|
import WidgetUtils from "../../src/utils/WidgetUtils";
|
|
|
|
import { WidgetType } from "../../src/widgets/WidgetType";
|
|
|
|
import { warnSelfDemote } from "../../src/components/views/right_panel/UserInfo";
|
|
|
|
import dispatcher from "../../src/dispatcher/dispatcher";
|
|
|
|
import { SettingLevel } from "../../src/settings/SettingLevel";
|
2023-07-11 12:53:33 +00:00
|
|
|
|
2024-10-15 13:57:26 +00:00
|
|
|
jest.mock("../../src/components/views/right_panel/UserInfo");
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2022-06-07 20:20:32 +00:00
|
|
|
describe("SlashCommands", () => {
|
|
|
|
let client: MatrixClient;
|
2022-08-16 13:20:26 +00:00
|
|
|
const roomId = "!room:example.com";
|
|
|
|
let room: Room;
|
|
|
|
const localRoomId = LOCAL_ROOM_ID_PREFIX + "test";
|
|
|
|
let localRoom: LocalRoom;
|
|
|
|
let command: Command;
|
|
|
|
|
2023-02-15 13:36:22 +00:00
|
|
|
const findCommand = (cmd: string): Command | undefined => {
|
2022-08-16 13:20:26 +00:00
|
|
|
return Commands.find((command: Command) => command.command === cmd);
|
|
|
|
};
|
|
|
|
|
|
|
|
const setCurrentRoom = (): void => {
|
2022-10-19 12:07:03 +00:00
|
|
|
mocked(SdkContextClass.instance.roomViewStore.getRoomId).mockReturnValue(roomId);
|
2023-02-15 13:36:22 +00:00
|
|
|
mocked(client.getRoom).mockImplementation((rId: string): Room | null => {
|
2022-08-16 13:20:26 +00:00
|
|
|
if (rId === roomId) return room;
|
2023-02-15 13:36:22 +00:00
|
|
|
return null;
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-07-11 12:53:33 +00:00
|
|
|
const setCurrentLocalRoom = (): void => {
|
2022-10-19 12:07:03 +00:00
|
|
|
mocked(SdkContextClass.instance.roomViewStore.getRoomId).mockReturnValue(localRoomId);
|
2023-02-15 13:36:22 +00:00
|
|
|
mocked(client.getRoom).mockImplementation((rId: string): Room | null => {
|
2022-08-16 13:20:26 +00:00
|
|
|
if (rId === localRoomId) return localRoom;
|
2023-02-15 13:36:22 +00:00
|
|
|
return null;
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
};
|
2022-06-07 20:20:32 +00:00
|
|
|
|
|
|
|
beforeEach(() => {
|
2022-08-16 13:20:26 +00:00
|
|
|
jest.clearAllMocks();
|
|
|
|
|
2022-06-07 20:20:32 +00:00
|
|
|
client = createTestClient();
|
2022-08-16 13:20:26 +00:00
|
|
|
|
2023-07-11 12:53:33 +00:00
|
|
|
room = new Room(roomId, client, client.getSafeUserId());
|
|
|
|
localRoom = new LocalRoom(localRoomId, client, client.getSafeUserId());
|
2022-08-16 13:20:26 +00:00
|
|
|
|
2022-10-19 12:07:03 +00:00
|
|
|
jest.spyOn(SdkContextClass.instance.roomViewStore, "getRoomId");
|
2022-06-07 20:20:32 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("/topic", () => {
|
|
|
|
it("sets topic", async () => {
|
|
|
|
const command = getCommand("/topic pizza");
|
|
|
|
expect(command.cmd).toBeDefined();
|
|
|
|
expect(command.args).toBeDefined();
|
2023-05-25 15:29:48 +00:00
|
|
|
await command.cmd!.run(client, "room-id", null, command.args);
|
2022-06-07 20:20:32 +00:00
|
|
|
expect(client.setRoomTopic).toHaveBeenCalledWith("room-id", "pizza", undefined);
|
|
|
|
});
|
2023-05-25 15:29:48 +00:00
|
|
|
|
|
|
|
it("should show topic modal if no args passed", async () => {
|
|
|
|
const spy = jest.spyOn(Modal, "createDialog");
|
|
|
|
const command = getCommand("/topic")!;
|
|
|
|
await command.cmd!.run(client, roomId, null);
|
|
|
|
expect(spy).toHaveBeenCalled();
|
|
|
|
});
|
2022-06-07 20:20:32 +00:00
|
|
|
});
|
2022-08-16 13:20:26 +00:00
|
|
|
|
|
|
|
describe.each([
|
|
|
|
["myroomnick"],
|
|
|
|
["roomavatar"],
|
|
|
|
["myroomavatar"],
|
|
|
|
["topic"],
|
|
|
|
["roomname"],
|
|
|
|
["invite"],
|
|
|
|
["part"],
|
|
|
|
["remove"],
|
|
|
|
["ban"],
|
|
|
|
["unban"],
|
|
|
|
["op"],
|
|
|
|
["deop"],
|
|
|
|
["addwidget"],
|
|
|
|
["discardsession"],
|
|
|
|
["whois"],
|
|
|
|
["holdcall"],
|
|
|
|
["unholdcall"],
|
|
|
|
["converttodm"],
|
|
|
|
["converttoroom"],
|
|
|
|
])("/%s", (commandName: string) => {
|
|
|
|
beforeEach(() => {
|
2023-02-15 13:36:22 +00:00
|
|
|
command = findCommand(commandName)!;
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("isEnabled", () => {
|
|
|
|
it("should return true for Room", () => {
|
|
|
|
setCurrentRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(true);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for LocalRoom", () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
setCurrentLocalRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-10-13 09:48:32 +00:00
|
|
|
describe("/upgraderoom", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
command = findCommand("upgraderoom")!;
|
|
|
|
setCurrentRoom();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should be disabled by default", () => {
|
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should be enabled for developerMode", () => {
|
|
|
|
SettingsStore.setValue("developerMode", null, SettingLevel.DEVICE, true);
|
|
|
|
expect(command.isEnabled(client)).toBe(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-07-11 12:53:33 +00:00
|
|
|
describe("/op", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
command = findCommand("op")!;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return usage if no args", () => {
|
|
|
|
expect(command.run(client, roomId, null, undefined).error).toBe(command.getUsage());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should reject with usage if given an invalid power level value", () => {
|
|
|
|
expect(command.run(client, roomId, null, "@bob:server Admin").error).toBe(command.getUsage());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should reject with usage for invalid input", () => {
|
|
|
|
expect(command.run(client, roomId, null, " ").error).toBe(command.getUsage());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should warn about self demotion", async () => {
|
|
|
|
setCurrentRoom();
|
|
|
|
const member = new RoomMember(roomId, client.getSafeUserId());
|
2024-03-12 14:52:54 +00:00
|
|
|
member.membership = KnownMembership.Join;
|
2023-07-11 12:53:33 +00:00
|
|
|
member.powerLevel = 100;
|
|
|
|
room.getMember = () => member;
|
|
|
|
command.run(client, roomId, null, `${client.getUserId()} 0`);
|
|
|
|
expect(warnSelfDemote).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should default to 50 if no powerlevel specified", async () => {
|
|
|
|
setCurrentRoom();
|
|
|
|
const member = new RoomMember(roomId, "@user:server");
|
2024-03-12 14:52:54 +00:00
|
|
|
member.membership = KnownMembership.Join;
|
2023-07-11 12:53:33 +00:00
|
|
|
room.getMember = () => member;
|
|
|
|
command.run(client, roomId, null, member.userId);
|
|
|
|
expect(client.setPowerLevel).toHaveBeenCalledWith(roomId, member.userId, 50);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("/deop", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
command = findCommand("deop")!;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return usage if no args", () => {
|
|
|
|
expect(command.run(client, roomId, null, undefined).error).toBe(command.getUsage());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should warn about self demotion", async () => {
|
|
|
|
setCurrentRoom();
|
|
|
|
const member = new RoomMember(roomId, client.getSafeUserId());
|
2024-03-12 14:52:54 +00:00
|
|
|
member.membership = KnownMembership.Join;
|
2023-07-11 12:53:33 +00:00
|
|
|
member.powerLevel = 100;
|
|
|
|
room.getMember = () => member;
|
|
|
|
command.run(client, roomId, null, client.getSafeUserId());
|
|
|
|
expect(warnSelfDemote).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should reject with usage for invalid input", () => {
|
|
|
|
expect(command.run(client, roomId, null, " ").error).toBe(command.getUsage());
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-08-16 13:20:26 +00:00
|
|
|
describe("/tovirtual", () => {
|
|
|
|
beforeEach(() => {
|
2023-02-15 13:36:22 +00:00
|
|
|
command = findCommand("tovirtual")!;
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("isEnabled", () => {
|
|
|
|
describe("when virtual rooms are supported", () => {
|
|
|
|
beforeEach(() => {
|
2022-08-30 19:13:39 +00:00
|
|
|
jest.spyOn(LegacyCallHandler.instance, "getSupportsVirtualRooms").mockReturnValue(true);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return true for Room", () => {
|
|
|
|
setCurrentRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(true);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for LocalRoom", () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
setCurrentLocalRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when virtual rooms are not supported", () => {
|
|
|
|
beforeEach(() => {
|
2022-08-30 19:13:39 +00:00
|
|
|
jest.spyOn(LegacyCallHandler.instance, "getSupportsVirtualRooms").mockReturnValue(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for Room", () => {
|
|
|
|
setCurrentRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for LocalRoom", () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
setCurrentLocalRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("/remakeolm", () => {
|
|
|
|
beforeEach(() => {
|
2023-02-15 13:36:22 +00:00
|
|
|
command = findCommand("remakeolm")!;
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("isEnabled", () => {
|
|
|
|
describe("when developer mode is enabled", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((settingName: string) => {
|
|
|
|
if (settingName === "developerMode") return true;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return true for Room", () => {
|
|
|
|
setCurrentRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(true);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for LocalRoom", () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
setCurrentLocalRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when developer mode is not enabled", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((settingName: string) => {
|
|
|
|
if (settingName === "developerMode") return false;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for Room", () => {
|
|
|
|
setCurrentRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false for LocalRoom", () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
setCurrentLocalRoom();
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.isEnabled(client)).toBe(false);
|
2022-08-16 13:20:26 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2022-11-07 13:45:34 +00:00
|
|
|
|
|
|
|
describe("/part", () => {
|
|
|
|
it("should part room matching alias if found", async () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
const room1 = new Room("room-id", client, client.getSafeUserId());
|
2022-11-07 13:45:34 +00:00
|
|
|
room1.getCanonicalAlias = jest.fn().mockReturnValue("#foo:bar");
|
2023-07-11 12:53:33 +00:00
|
|
|
const room2 = new Room("other-room", client, client.getSafeUserId());
|
2022-11-07 13:45:34 +00:00
|
|
|
room2.getCanonicalAlias = jest.fn().mockReturnValue("#baz:bar");
|
|
|
|
mocked(client.getRooms).mockReturnValue([room1, room2]);
|
|
|
|
|
|
|
|
const command = getCommand("/part #foo:bar");
|
|
|
|
expect(command.cmd).toBeDefined();
|
|
|
|
expect(command.args).toBeDefined();
|
2023-05-25 15:29:48 +00:00
|
|
|
await command.cmd!.run(client, "room-id", null, command.args);
|
2022-11-07 13:45:34 +00:00
|
|
|
expect(client.leaveRoomChain).toHaveBeenCalledWith("room-id", expect.anything());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should part room matching alt alias if found", async () => {
|
2023-07-11 12:53:33 +00:00
|
|
|
const room1 = new Room("room-id", client, client.getSafeUserId());
|
2022-11-07 13:45:34 +00:00
|
|
|
room1.getAltAliases = jest.fn().mockReturnValue(["#foo:bar"]);
|
2023-07-11 12:53:33 +00:00
|
|
|
const room2 = new Room("other-room", client, client.getSafeUserId());
|
2022-11-07 13:45:34 +00:00
|
|
|
room2.getAltAliases = jest.fn().mockReturnValue(["#baz:bar"]);
|
|
|
|
mocked(client.getRooms).mockReturnValue([room1, room2]);
|
|
|
|
|
|
|
|
const command = getCommand("/part #foo:bar");
|
|
|
|
expect(command.cmd).toBeDefined();
|
|
|
|
expect(command.args).toBeDefined();
|
2023-05-25 15:29:48 +00:00
|
|
|
await command.cmd!.run(client, "room-id", null, command.args!);
|
2022-11-07 13:45:34 +00:00
|
|
|
expect(client.leaveRoomChain).toHaveBeenCalledWith("room-id", expect.anything());
|
|
|
|
});
|
|
|
|
});
|
2022-11-21 11:24:59 +00:00
|
|
|
|
|
|
|
describe.each(["rainbow", "rainbowme"])("/%s", (commandName: string) => {
|
2023-02-15 13:36:22 +00:00
|
|
|
const command = findCommand(commandName)!;
|
2022-11-21 11:24:59 +00:00
|
|
|
|
|
|
|
it("should return usage if no args", () => {
|
2023-05-25 15:29:48 +00:00
|
|
|
expect(command.run(client, roomId, null, undefined).error).toBe(command.getUsage());
|
2022-11-21 11:24:59 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should make things rainbowy", () => {
|
2023-05-25 15:29:48 +00:00
|
|
|
return expect(
|
|
|
|
command.run(client, roomId, null, "this is a test message").promise,
|
|
|
|
).resolves.toMatchSnapshot();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe.each(["shrug", "tableflip", "unflip", "lenny"])("/%s", (commandName: string) => {
|
|
|
|
const command = findCommand(commandName)!;
|
|
|
|
|
|
|
|
it("should match snapshot with no args", () => {
|
|
|
|
return expect(command.run(client, roomId, null).promise).resolves.toMatchSnapshot();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should match snapshot with args", () => {
|
|
|
|
return expect(
|
|
|
|
command.run(client, roomId, null, "this is a test message").promise,
|
|
|
|
).resolves.toMatchSnapshot();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("/addwidget", () => {
|
|
|
|
it("should parse html iframe snippets", async () => {
|
|
|
|
jest.spyOn(WidgetUtils, "canUserModifyWidgets").mockReturnValue(true);
|
|
|
|
const spy = jest.spyOn(WidgetUtils, "setRoomWidget");
|
|
|
|
const command = findCommand("addwidget")!;
|
|
|
|
await command.run(client, roomId, null, '<iframe src="https://element.io"></iframe>');
|
|
|
|
expect(spy).toHaveBeenCalledWith(
|
|
|
|
client,
|
|
|
|
roomId,
|
|
|
|
expect.any(String),
|
|
|
|
WidgetType.CUSTOM,
|
|
|
|
"https://element.io",
|
|
|
|
"Custom",
|
|
|
|
{},
|
|
|
|
);
|
2022-11-21 11:24:59 +00:00
|
|
|
});
|
|
|
|
});
|
2023-07-14 11:20:59 +00:00
|
|
|
|
|
|
|
describe("/join", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.spyOn(dispatcher, "dispatch");
|
2024-03-12 14:52:54 +00:00
|
|
|
command = findCommand(KnownMembership.Join)!;
|
2023-07-14 11:20:59 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should return usage if no args", () => {
|
|
|
|
expect(command.run(client, roomId, null, undefined).error).toBe(command.getUsage());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle matrix.org permalinks", () => {
|
|
|
|
command.run(client, roomId, null, "https://matrix.to/#/!roomId:server/$eventId");
|
|
|
|
expect(dispatcher.dispatch).toHaveBeenCalledWith(
|
|
|
|
expect.objectContaining({
|
|
|
|
action: "view_room",
|
|
|
|
room_id: "!roomId:server",
|
|
|
|
event_id: "$eventId",
|
|
|
|
highlighted: true,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle room aliases", () => {
|
|
|
|
command.run(client, roomId, null, "#test:server");
|
|
|
|
expect(dispatcher.dispatch).toHaveBeenCalledWith(
|
|
|
|
expect.objectContaining({
|
|
|
|
action: "view_room",
|
|
|
|
room_alias: "#test:server",
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle room aliases with no server component", () => {
|
|
|
|
command.run(client, roomId, null, "#test");
|
|
|
|
expect(dispatcher.dispatch).toHaveBeenCalledWith(
|
|
|
|
expect.objectContaining({
|
|
|
|
action: "view_room",
|
|
|
|
room_alias: `#test:${client.getDomain()}`,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle room IDs and via servers", () => {
|
|
|
|
command.run(client, roomId, null, "!foo:bar serv1.com serv2.com");
|
|
|
|
expect(dispatcher.dispatch).toHaveBeenCalledWith(
|
|
|
|
expect.objectContaining({
|
|
|
|
action: "view_room",
|
|
|
|
room_id: "!foo:bar",
|
|
|
|
via_servers: ["serv1.com", "serv2.com"],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2022-06-07 20:20:32 +00:00
|
|
|
});
|