// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. #nullable disable using System.Collections.Generic; using System.Linq; using NUnit.Framework; using osu.Framework.Allocation; using osu.Framework.Graphics.Containers; using osu.Framework.Testing; using osu.Game.Online.API; using osu.Game.Online.API.Requests; using osu.Game.Online.API.Requests.Responses; using osu.Game.Online.Chat; using osu.Game.Tests.Visual; namespace osu.Game.Tests.Chat { [HeadlessTest] public partial class TestSceneChannelManager : OsuTestScene { private ChannelManager channelManager; private int currentMessageId; private List sentMessages; private List silencedUserIds; [SetUp] public void Setup() => Schedule(() => { var container = new ChannelManagerContainer(API); Child = container; channelManager = container.ChannelManager; }); [SetUpSteps] public void SetUpSteps() { AddStep("register request handling", () => { currentMessageId = 0; sentMessages = new List(); silencedUserIds = new List(); ((DummyAPIAccess)API).HandleRequest = req => { switch (req) { case JoinChannelRequest joinChannel: joinChannel.TriggerSuccess(); return true; case PostMessageRequest postMessage: handlePostMessageRequest(postMessage); return true; case MarkChannelAsReadRequest markRead: handleMarkChannelAsReadRequest(markRead); return true; case ChatAckRequest ack: ack.TriggerSuccess(new ChatAckResponse { Silences = silencedUserIds.Select(u => new ChatSilence { UserId = u }).ToList() }); silencedUserIds.Clear(); return true; case GetUpdatesRequest updatesRequest: updatesRequest.TriggerSuccess(new GetUpdatesResponse { Messages = sentMessages.ToList(), Presence = new List() }); return true; } return false; }; }); AddUntilStep("wait for notifications client", () => channelManager.NotificationsConnected); } [Test] public void TestCommandsPostedToCorrectChannelWhenNotCurrent() { Channel channel1 = null; Channel channel2 = null; AddStep("join 2 rooms", () => { channelManager.JoinChannel(channel1 = createChannel(1, ChannelType.Public)); channelManager.JoinChannel(channel2 = createChannel(2, ChannelType.Public)); }); AddStep("select channel 1", () => channelManager.CurrentChannel.Value = channel1); AddStep("post /me command to channel 2", () => channelManager.PostCommand("me dances", channel2)); AddAssert("/me command received by channel 2", () => channel2.Messages.Last().Content == "dances"); AddStep("post /np command to channel 2", () => channelManager.PostCommand("np", channel2)); AddAssert("/np command received by channel 2", () => channel2.Messages.Last().Content.Contains("is listening to")); } [Test] public void TestMarkAsReadIgnoringLocalMessages() { Channel channel = null; AddStep("join channel and select it", () => { channelManager.JoinChannel(channel = createChannel(1, ChannelType.Public)); channelManager.CurrentChannel.Value = channel; }); AddStep("post message", () => channelManager.PostMessage("Something interesting")); AddUntilStep("message posted", () => !channel.Messages.Any(m => m is LocalMessage)); AddStep("post /help command", () => channelManager.PostCommand("help", channel)); AddStep("post /me command with no action", () => channelManager.PostCommand("me", channel)); AddStep("post /join command with no channel", () => channelManager.PostCommand("join", channel)); AddStep("post /join command with non-existent channel", () => channelManager.PostCommand("join i-dont-exist", channel)); AddStep("post non-existent command", () => channelManager.PostCommand("non-existent-cmd arg", channel)); AddStep("mark channel as read", () => channelManager.MarkChannelAsRead(channel)); AddAssert("channel's last read ID is set to the latest message", () => channel.LastReadId == sentMessages.Last().Id); } [Test] public void TestSilencedUsersAreRemoved() { Channel channel = null; AddStep("join channel and select it", () => { channelManager.JoinChannel(channel = createChannel(1, ChannelType.Public)); channelManager.CurrentChannel.Value = channel; }); AddStep("post message", () => channelManager.PostMessage("Definitely something bad")); AddStep("mark user as silenced and send ack request", () => { silencedUserIds.Add(API.LocalUser.Value.OnlineID); channelManager.SendAck(); }); AddAssert("channel has no more messages", () => channel.Messages, () => Is.Empty); } [Test] public void TestCommandNameCaseInsensitivity() { Channel channel = null; AddStep("join channel and select it", () => { channelManager.JoinChannel(channel = createChannel(1, ChannelType.Public)); channelManager.CurrentChannel.Value = channel; }); AddStep("post /me command", () => channelManager.PostCommand("ME DANCES")); AddUntilStep("/me command received", () => channel.Messages.Last().Content.Contains("DANCES")); AddStep("post /help command", () => channelManager.PostCommand("HeLp")); AddUntilStep("/help command received", () => channel.Messages.Last().Content.Contains("Supported commands")); } private void handlePostMessageRequest(PostMessageRequest request) { var message = new Message(++currentMessageId) { IsAction = request.Message.IsAction, ChannelId = request.Message.ChannelId, Content = request.Message.Content, Links = request.Message.Links, Timestamp = request.Message.Timestamp, Sender = request.Message.Sender, Uuid = request.Message.Uuid }; sentMessages.Add(message); request.TriggerSuccess(message); } private void handleMarkChannelAsReadRequest(MarkChannelAsReadRequest request) { // only accept messages that were sent through the API if (sentMessages.Contains(request.Message)) { request.TriggerSuccess(); } else { request.TriggerFailure(new APIException("unknown message!", null)); } } private Channel createChannel(int id, ChannelType type) => new Channel(new APIUser()) { Id = id, Name = $"Channel {id}", Topic = $"Topic of channel {id} with type {type}", Type = type, LastMessageId = 0, }; private partial class ChannelManagerContainer : CompositeDrawable { [Cached] public ChannelManager ChannelManager { get; } public ChannelManagerContainer(IAPIProvider apiProvider) { InternalChild = ChannelManager = new ChannelManager(apiProvider); } } } }