mirror of
https://github.com/farcasclaudiu/openclaw.git
synced 2026-06-28 17:01:53 +03:00
perf(test): streamline imessage monitor suites
This commit is contained in:
@@ -60,7 +60,7 @@ async function closeMonitor() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
describe("monitorIMessageProvider", () => {
|
describe("monitorIMessageProvider", () => {
|
||||||
it("ignores malformed rpc message payloads", async () => {
|
it("handles default config gating, formatting, and reply context", async () => {
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
@@ -69,39 +69,104 @@ describe("monitorIMessageProvider", () => {
|
|||||||
sender: { nested: "not-a-string" },
|
sender: { nested: "not-a-string" },
|
||||||
text: "hello",
|
text: "hello",
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
await closeMonitor();
|
|
||||||
await run;
|
|
||||||
|
|
||||||
expect(replyMock).not.toHaveBeenCalled();
|
expect(replyMock).not.toHaveBeenCalled();
|
||||||
expect(sendMock).not.toHaveBeenCalled();
|
expect(sendMock).not.toHaveBeenCalled();
|
||||||
});
|
replyMock.mockClear();
|
||||||
|
sendMock.mockClear();
|
||||||
|
|
||||||
it("skips group messages without a mention by default", async () => {
|
notifyMessage({
|
||||||
const run = startMonitor();
|
id: 2,
|
||||||
await waitForSubscribe();
|
chat_id: 99,
|
||||||
|
sender: "+15550001111",
|
||||||
getNotificationHandler()?.({
|
is_from_me: false,
|
||||||
method: "message",
|
text: "hello group",
|
||||||
params: {
|
is_group: true,
|
||||||
message: {
|
|
||||||
id: 1,
|
|
||||||
chat_id: 99,
|
|
||||||
sender: "+15550001111",
|
|
||||||
is_from_me: false,
|
|
||||||
text: "hello group",
|
|
||||||
is_group: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
await closeMonitor();
|
|
||||||
await run;
|
|
||||||
|
|
||||||
expect(replyMock).not.toHaveBeenCalled();
|
expect(replyMock).not.toHaveBeenCalled();
|
||||||
expect(sendMock).not.toHaveBeenCalled();
|
expect(sendMock).not.toHaveBeenCalled();
|
||||||
|
replyMock.mockClear();
|
||||||
|
sendMock.mockClear();
|
||||||
|
|
||||||
|
replyMock.mockResolvedValueOnce({ text: "yo" });
|
||||||
|
notifyMessage({
|
||||||
|
id: 3,
|
||||||
|
chat_id: 42,
|
||||||
|
sender: "+15550002222",
|
||||||
|
is_from_me: false,
|
||||||
|
text: "@openclaw ping",
|
||||||
|
is_group: true,
|
||||||
|
chat_name: "Lobster Squad",
|
||||||
|
participants: ["+1555", "+1556"],
|
||||||
|
});
|
||||||
|
await flush();
|
||||||
|
expect(replyMock).toHaveBeenCalledOnce();
|
||||||
|
{
|
||||||
|
const ctx = replyMock.mock.calls[0]?.[0] as { Body?: string; ChatType?: string };
|
||||||
|
expect(ctx.ChatType).toBe("group");
|
||||||
|
// Sender should appear as prefix in group messages (no redundant [from:] suffix)
|
||||||
|
expect(String(ctx.Body ?? "")).toContain("+15550002222:");
|
||||||
|
expect(String(ctx.Body ?? "")).not.toContain("[from:");
|
||||||
|
}
|
||||||
|
expect(sendMock).toHaveBeenCalledWith(
|
||||||
|
"chat_id:42",
|
||||||
|
"yo",
|
||||||
|
expect.objectContaining({ client: expect.any(Object) }),
|
||||||
|
);
|
||||||
|
replyMock.mockClear();
|
||||||
|
sendMock.mockClear();
|
||||||
|
|
||||||
|
notifyMessage({
|
||||||
|
id: 4,
|
||||||
|
chat_id: 99,
|
||||||
|
chat_name: "Test Group",
|
||||||
|
sender: "+15550001111",
|
||||||
|
is_from_me: false,
|
||||||
|
text: "@openclaw hi",
|
||||||
|
is_group: true,
|
||||||
|
created_at: "2026-01-17T00:00:00Z",
|
||||||
|
});
|
||||||
|
await flush();
|
||||||
|
expect(replyMock).toHaveBeenCalled();
|
||||||
|
{
|
||||||
|
const ctx = replyMock.mock.calls[0]?.[0];
|
||||||
|
const body = ctx?.Body ?? "";
|
||||||
|
expect(body).toContain("Test Group id:99");
|
||||||
|
expect(body).toContain("+15550001111: @openclaw hi");
|
||||||
|
}
|
||||||
|
replyMock.mockClear();
|
||||||
|
sendMock.mockClear();
|
||||||
|
|
||||||
|
notifyMessage({
|
||||||
|
id: 5,
|
||||||
|
chat_id: 55,
|
||||||
|
sender: "+15550001111",
|
||||||
|
is_from_me: false,
|
||||||
|
text: "replying now",
|
||||||
|
is_group: false,
|
||||||
|
reply_to_id: 9001,
|
||||||
|
reply_to_text: "original message",
|
||||||
|
reply_to_sender: "+15559998888",
|
||||||
|
});
|
||||||
|
await flush();
|
||||||
|
expect(replyMock).toHaveBeenCalled();
|
||||||
|
{
|
||||||
|
const ctx = replyMock.mock.calls[0]?.[0] as {
|
||||||
|
Body?: string;
|
||||||
|
ReplyToId?: string;
|
||||||
|
ReplyToBody?: string;
|
||||||
|
ReplyToSender?: string;
|
||||||
|
};
|
||||||
|
expect(ctx.ReplyToId).toBe("9001");
|
||||||
|
expect(ctx.ReplyToBody).toBe("original message");
|
||||||
|
expect(ctx.ReplyToSender).toBe("+15559998888");
|
||||||
|
expect(String(ctx.Body ?? "")).toContain("[Replying to +15559998888 id:9001]");
|
||||||
|
expect(String(ctx.Body ?? "")).toContain("original message");
|
||||||
|
}
|
||||||
|
|
||||||
|
await closeMonitor();
|
||||||
|
await run;
|
||||||
});
|
});
|
||||||
|
|
||||||
it("allows group messages when imessage groups default disables mention gating", async () => {
|
it("allows group messages when imessage groups default disables mention gating", async () => {
|
||||||
@@ -117,21 +182,17 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 11,
|
||||||
params: {
|
chat_id: 123,
|
||||||
message: {
|
sender: "+15550001111",
|
||||||
id: 11,
|
is_from_me: false,
|
||||||
chat_id: 123,
|
text: "hello group",
|
||||||
sender: "+15550001111",
|
is_group: true,
|
||||||
is_from_me: false,
|
|
||||||
text: "hello group",
|
|
||||||
is_group: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -145,7 +206,10 @@ describe("monitorIMessageProvider", () => {
|
|||||||
const config = getConfig();
|
const config = getConfig();
|
||||||
setConfigMock({
|
setConfigMock({
|
||||||
...config,
|
...config,
|
||||||
messages: { groupChat: { mentionPatterns: [] } },
|
messages: {
|
||||||
|
...config.messages,
|
||||||
|
groupChat: { mentionPatterns: [] },
|
||||||
|
},
|
||||||
channels: {
|
channels: {
|
||||||
...config.channels,
|
...config.channels,
|
||||||
imessage: {
|
imessage: {
|
||||||
@@ -155,21 +219,17 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 12,
|
||||||
params: {
|
chat_id: 777,
|
||||||
message: {
|
sender: "+15550001111",
|
||||||
id: 12,
|
is_from_me: false,
|
||||||
chat_id: 777,
|
text: "hello group",
|
||||||
sender: "+15550001111",
|
is_group: true,
|
||||||
is_from_me: false,
|
|
||||||
text: "hello group",
|
|
||||||
is_group: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -191,21 +251,17 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 13,
|
||||||
params: {
|
chat_id: 123,
|
||||||
message: {
|
sender: "+15550001111",
|
||||||
id: 13,
|
is_from_me: false,
|
||||||
chat_id: 123,
|
text: "@openclaw hello",
|
||||||
sender: "+15550001111",
|
is_group: true,
|
||||||
is_from_me: false,
|
|
||||||
text: "@openclaw hello",
|
|
||||||
is_group: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -234,18 +290,13 @@ describe("monitorIMessageProvider", () => {
|
|||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 14,
|
||||||
params: {
|
chat_id: 2,
|
||||||
message: {
|
sender: "+15550001111",
|
||||||
id: 14,
|
is_from_me: false,
|
||||||
chat_id: 2,
|
text: "hello",
|
||||||
sender: "+15550001111",
|
is_group: false,
|
||||||
is_from_me: false,
|
|
||||||
text: "hello",
|
|
||||||
is_group: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -265,24 +316,23 @@ describe("monitorIMessageProvider", () => {
|
|||||||
const config = getConfig();
|
const config = getConfig();
|
||||||
setConfigMock({
|
setConfigMock({
|
||||||
...config,
|
...config,
|
||||||
messages: { responsePrefix: "PFX" },
|
messages: {
|
||||||
|
...config.messages,
|
||||||
|
responsePrefix: "PFX",
|
||||||
|
},
|
||||||
});
|
});
|
||||||
replyMock.mockResolvedValue({ text: "final reply" });
|
replyMock.mockResolvedValue({ text: "final reply" });
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 7,
|
||||||
params: {
|
chat_id: 77,
|
||||||
message: {
|
sender: "+15550001111",
|
||||||
id: 7,
|
is_from_me: false,
|
||||||
chat_id: 77,
|
text: "hello",
|
||||||
sender: "+15550001111",
|
is_group: false,
|
||||||
is_from_me: false,
|
|
||||||
text: "hello",
|
|
||||||
is_group: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -307,21 +357,17 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 99,
|
||||||
params: {
|
chat_id: 77,
|
||||||
message: {
|
sender: "+15550001111",
|
||||||
id: 99,
|
is_from_me: false,
|
||||||
chat_id: 77,
|
text: "hello",
|
||||||
sender: "+15550001111",
|
is_group: false,
|
||||||
is_from_me: false,
|
|
||||||
text: "hello",
|
|
||||||
is_group: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -337,45 +383,6 @@ describe("monitorIMessageProvider", () => {
|
|||||||
expect(String(sendMock.mock.calls[0]?.[1] ?? "")).toContain("Pairing code: PAIRCODE");
|
expect(String(sendMock.mock.calls[0]?.[1] ?? "")).toContain("Pairing code: PAIRCODE");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("delivers group replies when mentioned", async () => {
|
|
||||||
replyMock.mockResolvedValueOnce({ text: "yo" });
|
|
||||||
const run = startMonitor();
|
|
||||||
await waitForSubscribe();
|
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
|
||||||
method: "message",
|
|
||||||
params: {
|
|
||||||
message: {
|
|
||||||
id: 2,
|
|
||||||
chat_id: 42,
|
|
||||||
sender: "+15550002222",
|
|
||||||
is_from_me: false,
|
|
||||||
text: "@openclaw ping",
|
|
||||||
is_group: true,
|
|
||||||
chat_name: "Lobster Squad",
|
|
||||||
participants: ["+1555", "+1556"],
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await flush();
|
|
||||||
await closeMonitor();
|
|
||||||
await run;
|
|
||||||
|
|
||||||
expect(replyMock).toHaveBeenCalledOnce();
|
|
||||||
const ctx = replyMock.mock.calls[0]?.[0] as { Body?: string; ChatType?: string };
|
|
||||||
expect(ctx.ChatType).toBe("group");
|
|
||||||
// Sender should appear as prefix in group messages (no redundant [from:] suffix)
|
|
||||||
expect(String(ctx.Body ?? "")).toContain("+15550002222:");
|
|
||||||
expect(String(ctx.Body ?? "")).not.toContain("[from:");
|
|
||||||
|
|
||||||
expect(sendMock).toHaveBeenCalledWith(
|
|
||||||
"chat_id:42",
|
|
||||||
"yo",
|
|
||||||
expect.objectContaining({ client: expect.any(Object) }),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("honors group allowlist when groupPolicy is allowlist", async () => {
|
it("honors group allowlist when groupPolicy is allowlist", async () => {
|
||||||
const config = getConfig();
|
const config = getConfig();
|
||||||
setConfigMock({
|
setConfigMock({
|
||||||
@@ -389,21 +396,17 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 3,
|
||||||
params: {
|
chat_id: 202,
|
||||||
message: {
|
sender: "+15550003333",
|
||||||
id: 3,
|
is_from_me: false,
|
||||||
chat_id: 202,
|
text: "@openclaw hi",
|
||||||
sender: "+15550003333",
|
is_group: true,
|
||||||
is_from_me: false,
|
|
||||||
text: "@openclaw hi",
|
|
||||||
is_group: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -430,6 +433,7 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
readAllowFromStoreMock.mockResolvedValue(["+15550003333"]);
|
readAllowFromStoreMock.mockResolvedValue(["+15550003333"]);
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
@@ -466,6 +470,7 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
readAllowFromStoreMock.mockResolvedValue(["+15550003333"]);
|
readAllowFromStoreMock.mockResolvedValue(["+15550003333"]);
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
@@ -502,6 +507,7 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
readAllowFromStoreMock.mockResolvedValue(["+15550003333"]);
|
readAllowFromStoreMock.mockResolvedValue(["+15550003333"]);
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
@@ -534,21 +540,17 @@ describe("monitorIMessageProvider", () => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const run = startMonitor();
|
const run = startMonitor();
|
||||||
await waitForSubscribe();
|
await waitForSubscribe();
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
notifyMessage({
|
||||||
method: "message",
|
id: 10,
|
||||||
params: {
|
chat_id: 303,
|
||||||
message: {
|
sender: "+15550003333",
|
||||||
id: 10,
|
is_from_me: false,
|
||||||
chat_id: 303,
|
text: "@openclaw hi",
|
||||||
sender: "+15550003333",
|
is_group: true,
|
||||||
is_from_me: false,
|
|
||||||
text: "@openclaw hi",
|
|
||||||
is_group: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await flush();
|
await flush();
|
||||||
@@ -557,74 +559,4 @@ describe("monitorIMessageProvider", () => {
|
|||||||
|
|
||||||
expect(replyMock).not.toHaveBeenCalled();
|
expect(replyMock).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("prefixes group message bodies with sender", async () => {
|
|
||||||
const run = startMonitor();
|
|
||||||
await waitForSubscribe();
|
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
|
||||||
method: "message",
|
|
||||||
params: {
|
|
||||||
message: {
|
|
||||||
id: 11,
|
|
||||||
chat_id: 99,
|
|
||||||
chat_name: "Test Group",
|
|
||||||
sender: "+15550001111",
|
|
||||||
is_from_me: false,
|
|
||||||
text: "@openclaw hi",
|
|
||||||
is_group: true,
|
|
||||||
created_at: "2026-01-17T00:00:00Z",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await flush();
|
|
||||||
await closeMonitor();
|
|
||||||
await run;
|
|
||||||
|
|
||||||
expect(replyMock).toHaveBeenCalled();
|
|
||||||
const ctx = replyMock.mock.calls[0]?.[0];
|
|
||||||
const body = ctx?.Body ?? "";
|
|
||||||
expect(body).toContain("Test Group id:99");
|
|
||||||
expect(body).toContain("+15550001111: @openclaw hi");
|
|
||||||
});
|
|
||||||
|
|
||||||
it("includes reply context when imessage reply metadata is present", async () => {
|
|
||||||
const run = startMonitor();
|
|
||||||
await waitForSubscribe();
|
|
||||||
|
|
||||||
getNotificationHandler()?.({
|
|
||||||
method: "message",
|
|
||||||
params: {
|
|
||||||
message: {
|
|
||||||
id: 12,
|
|
||||||
chat_id: 55,
|
|
||||||
sender: "+15550001111",
|
|
||||||
is_from_me: false,
|
|
||||||
text: "replying now",
|
|
||||||
is_group: false,
|
|
||||||
reply_to_id: 9001,
|
|
||||||
reply_to_text: "original message",
|
|
||||||
reply_to_sender: "+15559998888",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await flush();
|
|
||||||
await closeMonitor();
|
|
||||||
await run;
|
|
||||||
|
|
||||||
expect(replyMock).toHaveBeenCalled();
|
|
||||||
const ctx = replyMock.mock.calls[0]?.[0] as {
|
|
||||||
Body?: string;
|
|
||||||
ReplyToId?: string;
|
|
||||||
ReplyToBody?: string;
|
|
||||||
ReplyToSender?: string;
|
|
||||||
};
|
|
||||||
expect(ctx.ReplyToId).toBe("9001");
|
|
||||||
expect(ctx.ReplyToBody).toBe("original message");
|
|
||||||
expect(ctx.ReplyToSender).toBe("+15559998888");
|
|
||||||
expect(String(ctx.Body ?? "")).toContain("[Replying to +15559998888 id:9001]");
|
|
||||||
expect(String(ctx.Body ?? "")).toContain("original message");
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -112,12 +112,20 @@ vi.mock("./probe.js", () => ({
|
|||||||
export const flush = () => new Promise((resolve) => setTimeout(resolve, 0));
|
export const flush = () => new Promise((resolve) => setTimeout(resolve, 0));
|
||||||
|
|
||||||
export async function waitForSubscribe() {
|
export async function waitForSubscribe() {
|
||||||
|
for (let i = 0; i < 25; i += 1) {
|
||||||
|
if (state.requestMock.mock.calls.some((call) => call[0] === "watch.subscribe")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Prefer microtask turns over timers for speed.
|
||||||
|
await Promise.resolve();
|
||||||
|
}
|
||||||
for (let i = 0; i < 5; i += 1) {
|
for (let i = 0; i < 5; i += 1) {
|
||||||
if (state.requestMock.mock.calls.some((call) => call[0] === "watch.subscribe")) {
|
if (state.requestMock.mock.calls.some((call) => call[0] === "watch.subscribe")) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
await flush();
|
await flush();
|
||||||
}
|
}
|
||||||
|
throw new Error("imessage test harness: watch.subscribe not observed");
|
||||||
}
|
}
|
||||||
|
|
||||||
export function installMonitorIMessageProviderTestHooks() {
|
export function installMonitorIMessageProviderTestHooks() {
|
||||||
|
|||||||
Reference in New Issue
Block a user