2019-01-24 08:43:03 +00:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-14 11:32:48 +00:00
2022-06-17 07:37:17 +00:00
#nullable disable
2018-04-14 11:32:48 +00:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using osu.Framework.Allocation ;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables ;
2022-01-03 08:31:12 +00:00
using osu.Framework.Extensions ;
2022-10-28 07:22:35 +00:00
using osu.Framework.Graphics.Containers ;
2018-04-14 11:32:48 +00:00
using osu.Framework.Logging ;
2022-10-28 07:22:35 +00:00
using osu.Framework.Threading ;
2020-12-20 19:18:00 +00:00
using osu.Game.Database ;
2018-04-14 11:32:48 +00:00
using osu.Game.Online.API ;
using osu.Game.Online.API.Requests ;
2021-11-04 09:02:44 +00:00
using osu.Game.Online.API.Requests.Responses ;
2022-10-28 07:22:35 +00:00
using osu.Game.Online.Notifications ;
2022-05-19 10:26:14 +00:00
using osu.Game.Overlays.Chat.Listing ;
2018-04-14 11:32:48 +00:00
namespace osu.Game.Online.Chat
{
/// <summary>
/// Manages everything channel related
/// </summary>
2022-10-28 07:22:35 +00:00
public partial class ChannelManager : CompositeComponent , IChannelPostTarget
2018-04-14 11:32:48 +00:00
{
/// <summary>
/// The channels the player joins on startup
/// </summary>
private readonly string [ ] defaultChannels =
{
@"#lazer" ,
@"#osu" ,
@"#lobby"
} ;
2019-01-07 09:50:27 +00:00
private readonly BindableList < Channel > availableChannels = new BindableList < Channel > ( ) ;
private readonly BindableList < Channel > joinedChannels = new BindableList < Channel > ( ) ;
2018-11-21 18:15:55 +00:00
2020-12-17 22:56:34 +00:00
/// <summary>
2020-12-20 19:18:00 +00:00
/// Keeps a stack of recently closed channels
2020-12-17 22:56:34 +00:00
/// </summary>
2021-06-03 11:59:25 +00:00
private readonly List < ClosedChannel > closedChannels = new List < ClosedChannel > ( ) ;
2020-12-13 18:21:50 +00:00
2020-12-14 01:46:02 +00:00
// For efficiency purposes, this constant bounds the number of closed channels we store.
// This number is somewhat arbitrary; future developers are free to modify it.
// Must be a positive number.
2020-12-14 04:27:48 +00:00
private const int closed_channels_max_size = 50 ;
2020-12-14 01:46:02 +00:00
2018-04-14 11:32:48 +00:00
/// <summary>
/// The currently opened channel
/// </summary>
public Bindable < Channel > CurrentChannel { get ; } = new Bindable < Channel > ( ) ;
/// <summary>
/// The Channels the player has joined
/// </summary>
2019-01-07 09:50:27 +00:00
public IBindableList < Channel > JoinedChannels = > joinedChannels ;
2018-04-14 11:32:48 +00:00
/// <summary>
/// The channels available for the player to join
/// </summary>
2019-01-07 09:50:27 +00:00
public IBindableList < Channel > AvailableChannels = > availableChannels ;
2018-04-14 11:32:48 +00:00
2023-01-12 20:14:21 +00:00
/// <summary>
/// Whether the client responsible for channel notifications is connected.
/// </summary>
public bool NotificationsConnected = > connector . IsConnected . Value ;
2022-06-08 09:54:23 +00:00
private readonly IAPIProvider api ;
2022-10-28 09:08:08 +00:00
private readonly NotificationsClientConnector connector ;
2018-12-10 12:08:14 +00:00
2020-12-20 19:18:00 +00:00
[Resolved]
private UserLookupCache users { get ; set ; }
2022-10-28 07:22:35 +00:00
private readonly IBindable < APIState > apiState = new Bindable < APIState > ( ) ;
2022-11-04 10:02:26 +00:00
private ScheduledDelegate scheduledAck ;
2021-12-26 07:26:47 +00:00
2022-11-12 14:32:05 +00:00
private long? lastSilenceMessageId ;
2022-11-02 08:13:14 +00:00
private uint? lastSilenceId ;
2022-11-04 09:51:00 +00:00
public ChannelManager ( IAPIProvider api )
2018-09-14 03:06:04 +00:00
{
2022-06-08 09:54:23 +00:00
this . api = api ;
2022-11-04 09:51:00 +00:00
connector = api . GetNotificationsConnector ( ) ;
2022-10-28 09:08:08 +00:00
2018-09-14 03:06:04 +00:00
CurrentChannel . ValueChanged + = currentChannelChanged ;
2021-12-26 07:26:47 +00:00
}
2022-10-28 07:22:35 +00:00
[BackgroundDependencyLoader]
private void load ( )
2021-12-26 07:26:47 +00:00
{
2022-10-28 09:37:43 +00:00
connector . ChannelJoined + = ch = > Schedule ( ( ) = > joinChannel ( ch ) ) ;
2022-11-04 07:42:59 +00:00
2023-02-08 02:31:28 +00:00
connector . ChannelParted + = ch = > Schedule ( ( ) = > leaveChannel ( getChannel ( ch ) , false ) ) ;
2022-11-07 02:36:55 +00:00
2022-10-28 09:37:43 +00:00
connector . NewMessages + = msgs = > Schedule ( ( ) = > addMessages ( msgs ) ) ;
2022-11-04 07:42:59 +00:00
2023-01-09 07:38:37 +00:00
connector . PresenceReceived + = ( ) = > Schedule ( initializeChannels ) ;
2022-01-13 08:33:55 +00:00
2022-11-04 10:36:24 +00:00
connector . Start ( ) ;
2022-01-13 08:33:55 +00:00
2022-10-28 07:22:35 +00:00
apiState . BindTo ( api . State ) ;
2022-11-12 14:02:37 +00:00
apiState . BindValueChanged ( _ = > SendAck ( ) , true ) ;
2018-09-14 03:06:04 +00:00
}
2018-07-24 02:54:11 +00:00
/// <summary>
/// Opens a channel or switches to the channel if already opened.
/// </summary>
/// <exception cref="ChannelNotFoundException">If the name of the specifed channel was not found this exception will be thrown.</exception>
/// <param name="name"></param>
2018-04-14 11:32:48 +00:00
public void OpenChannel ( string name )
{
2022-12-22 20:27:59 +00:00
ArgumentNullException . ThrowIfNull ( name ) ;
2018-04-14 11:32:48 +00:00
2018-11-14 04:19:20 +00:00
CurrentChannel . Value = AvailableChannels . FirstOrDefault ( c = > c . Name = = name ) ? ? throw new ChannelNotFoundException ( name ) ;
2018-04-14 11:32:48 +00:00
}
2018-07-24 02:54:11 +00:00
/// <summary>
/// Opens a new private channel.
/// </summary>
2018-07-24 03:14:33 +00:00
/// <param name="user">The user the private channel is opened with.</param>
2021-11-04 09:02:44 +00:00
public void OpenPrivateChannel ( APIUser user )
2018-04-14 11:32:48 +00:00
{
2022-12-22 20:27:59 +00:00
ArgumentNullException . ThrowIfNull ( user ) ;
2018-04-14 11:32:48 +00:00
2019-06-03 09:25:19 +00:00
if ( user . Id = = api . LocalUser . Value . Id )
return ;
2018-11-13 06:20:40 +00:00
CurrentChannel . Value = JoinedChannels . FirstOrDefault ( c = > c . Type = = ChannelType . PM & & c . Users . Count = = 1 & & c . Users . Any ( u = > u . Id = = user . Id ) )
2020-06-08 08:49:45 +00:00
? ? JoinChannel ( new Channel ( user ) ) ;
2018-04-14 11:32:48 +00:00
}
2022-06-28 04:58:35 +00:00
private void currentChannelChanged ( ValueChangedEvent < Channel > channel )
2019-05-11 23:13:48 +00:00
{
2022-06-28 04:58:35 +00:00
bool isSelectorChannel = channel . NewValue is ChannelListing . ChannelListingChannel ;
2022-05-15 18:38:37 +00:00
if ( ! isSelectorChannel )
2022-06-28 04:58:35 +00:00
JoinChannel ( channel . NewValue ) ;
Logger . Log ( $"Current channel changed to {channel.NewValue}" ) ;
2019-05-11 23:13:48 +00:00
}
2018-11-13 08:24:11 +00:00
/// <summary>
/// Ensure we run post actions in sequence, once at a time.
/// </summary>
private readonly Queue < Action > postQueue = new Queue < Action > ( ) ;
2018-04-14 11:32:48 +00:00
/// <summary>
/// Posts a message to the currently opened channel.
/// </summary>
/// <param name="text">The message text that is going to be posted</param>
/// <param name="isAction">Is true if the message is an action, e.g.: user is currently eating </param>
2018-12-20 08:01:08 +00:00
/// <param name="target">An optional target channel. If null, <see cref="CurrentChannel"/> will be used.</param>
public void PostMessage ( string text , bool isAction = false , Channel target = null )
2018-04-14 11:32:48 +00:00
{
2020-06-03 07:48:44 +00:00
target ? ? = CurrentChannel . Value ;
2018-04-14 11:32:48 +00:00
2018-12-20 08:01:08 +00:00
if ( target = = null )
return ;
2018-07-09 19:00:39 +00:00
2018-11-13 08:24:11 +00:00
void dequeueAndRun ( )
2018-04-14 11:32:48 +00:00
{
2018-11-13 08:24:11 +00:00
if ( postQueue . Count > 0 )
postQueue . Dequeue ( ) . Invoke ( ) ;
2018-04-14 11:32:48 +00:00
}
2018-11-13 08:24:11 +00:00
postQueue . Enqueue ( ( ) = >
2018-04-14 11:32:48 +00:00
{
2018-11-13 08:24:11 +00:00
if ( ! api . IsLoggedIn )
{
2018-12-20 08:01:08 +00:00
target . AddNewMessages ( new ErrorMessage ( "Please sign in to participate in chat!" ) ) ;
2018-11-13 08:24:11 +00:00
return ;
}
2018-04-14 11:32:48 +00:00
2018-11-13 08:24:11 +00:00
var message = new LocalEchoMessage
{
Sender = api . LocalUser . Value ,
Timestamp = DateTimeOffset . Now ,
2018-12-20 08:01:08 +00:00
ChannelId = target . Id ,
2018-11-13 08:24:11 +00:00
IsAction = isAction ,
2022-11-04 07:42:59 +00:00
Content = text ,
Uuid = Guid . NewGuid ( ) . ToString ( )
2018-11-13 08:24:11 +00:00
} ;
2018-04-14 11:32:48 +00:00
2018-12-20 08:01:08 +00:00
target . AddLocalEcho ( message ) ;
2018-11-13 08:24:11 +00:00
// if this is a PM and the first message, we need to do a special request to create the PM channel
2020-06-08 08:49:45 +00:00
if ( target . Type = = ChannelType . PM & & target . Id = = 0 )
2018-11-13 08:24:11 +00:00
{
2018-12-20 08:01:08 +00:00
var createNewPrivateMessageRequest = new CreateNewPrivateMessageRequest ( target . Users . First ( ) , message ) ;
2018-11-14 04:59:02 +00:00
2022-11-04 07:42:59 +00:00
createNewPrivateMessageRequest . Success + = _ = > dequeueAndRun ( ) ;
2018-11-13 08:24:11 +00:00
createNewPrivateMessageRequest . Failure + = exception = >
{
2022-05-18 00:47:23 +00:00
handlePostException ( exception ) ;
2018-12-20 08:01:08 +00:00
target . ReplaceMessage ( message , null ) ;
2018-11-13 08:24:11 +00:00
dequeueAndRun ( ) ;
} ;
api . Queue ( createNewPrivateMessageRequest ) ;
return ;
}
var req = new PostMessageRequest ( message ) ;
2018-11-14 04:59:02 +00:00
2022-11-04 07:42:59 +00:00
req . Success + = m = > dequeueAndRun ( ) ;
2018-11-13 08:24:11 +00:00
req . Failure + = exception = >
{
2021-02-26 04:37:58 +00:00
handlePostException ( exception ) ;
2018-12-20 08:01:08 +00:00
target . ReplaceMessage ( message , null ) ;
2018-11-13 08:24:11 +00:00
dequeueAndRun ( ) ;
} ;
2018-11-14 04:59:02 +00:00
2018-11-13 08:24:11 +00:00
api . Queue ( req ) ;
} ) ;
// always run if the queue is empty
if ( postQueue . Count = = 1 )
dequeueAndRun ( ) ;
2018-04-14 11:32:48 +00:00
}
2021-02-26 04:37:58 +00:00
private static void handlePostException ( Exception exception )
{
if ( exception is APIException apiException )
Logger . Log ( apiException . Message , level : LogLevel . Important ) ;
else
Logger . Error ( exception , "Posting message failed." ) ;
}
2018-07-24 03:14:33 +00:00
/// <summary>
/// Posts a command locally. Commands like /help will result in a help message written in the current channel.
/// </summary>
/// <param name="text">the text containing the command identifier and command parameters.</param>
2018-12-20 08:01:08 +00:00
/// <param name="target">An optional target channel. If null, <see cref="CurrentChannel"/> will be used.</param>
public void PostCommand ( string text , Channel target = null )
2018-04-14 11:32:48 +00:00
{
2020-06-03 07:48:44 +00:00
target ? ? = CurrentChannel . Value ;
2018-12-20 08:01:08 +00:00
if ( target = = null )
2018-04-14 11:32:48 +00:00
return ;
2021-10-27 04:04:41 +00:00
string [ ] parameters = text . Split ( ' ' , 2 ) ;
2018-04-14 11:32:48 +00:00
string command = parameters [ 0 ] ;
string content = parameters . Length = = 2 ? parameters [ 1 ] : string . Empty ;
switch ( command )
{
2020-04-19 06:12:36 +00:00
case "np" :
2021-06-29 06:58:07 +00:00
AddInternal ( new NowPlayingCommand ( target ) ) ;
2020-04-19 06:12:36 +00:00
break ;
2018-04-14 11:32:48 +00:00
case "me" :
if ( string . IsNullOrWhiteSpace ( content ) )
{
2018-12-20 08:01:08 +00:00
target . AddNewMessages ( new ErrorMessage ( "Usage: /me [action]" ) ) ;
2018-04-14 11:32:48 +00:00
break ;
}
2021-06-29 06:58:07 +00:00
PostMessage ( content , true , target ) ;
2018-04-14 11:32:48 +00:00
break ;
2019-08-04 23:02:42 +00:00
case "join" :
if ( string . IsNullOrWhiteSpace ( content ) )
{
target . AddNewMessages ( new ErrorMessage ( "Usage: /join [channel]" ) ) ;
break ;
}
2019-11-11 12:05:36 +00:00
var channel = availableChannels . FirstOrDefault ( c = > c . Name = = content | | c . Name = = $"#{content}" ) ;
2019-08-08 07:02:09 +00:00
2019-08-04 23:02:42 +00:00
if ( channel = = null )
{
target . AddNewMessages ( new ErrorMessage ( $"Channel '{content}' not found." ) ) ;
break ;
}
JoinChannel ( channel ) ;
break ;
2021-09-05 17:16:57 +00:00
case "chat" :
2021-09-07 16:25:47 +00:00
case "msg" :
2021-09-05 20:10:08 +00:00
case "query" :
2021-09-05 17:16:57 +00:00
if ( string . IsNullOrWhiteSpace ( content ) )
{
2021-09-05 20:20:19 +00:00
target . AddNewMessages ( new ErrorMessage ( $"Usage: /{command} [user]" ) ) ;
2021-09-05 17:16:57 +00:00
break ;
}
2021-09-12 07:50:53 +00:00
// Check if the user has joined the requested channel already.
// This uses the channel name for comparison as the PM user's username is unavailable after a restart.
2021-09-10 07:15:43 +00:00
var privateChannel = JoinedChannels . FirstOrDefault (
2021-09-07 16:22:59 +00:00
c = > c . Type = = ChannelType . PM & & c . Users . Count = = 1 & & c . Name . Equals ( content , StringComparison . OrdinalIgnoreCase ) ) ;
2021-09-10 07:01:38 +00:00
2021-09-10 07:15:43 +00:00
if ( privateChannel ! = null )
2021-09-10 07:01:38 +00:00
{
2021-09-10 07:15:43 +00:00
CurrentChannel . Value = privateChannel ;
2021-09-10 07:01:38 +00:00
break ;
}
2021-09-07 16:06:12 +00:00
2021-09-05 17:16:57 +00:00
var request = new GetUserRequest ( content ) ;
request . Success + = OpenPrivateChannel ;
2021-09-11 13:18:09 +00:00
request . Failure + = e = > target . AddNewMessages (
2021-09-12 10:56:36 +00:00
new ErrorMessage ( e . InnerException ? . Message = = @"NotFound" ? $"User '{content}' was not found." : $"Could not fetch user '{content}'." ) ) ;
2021-09-11 13:18:09 +00:00
2021-09-05 17:16:57 +00:00
api . Queue ( request ) ;
break ;
2018-04-14 11:32:48 +00:00
case "help" :
2021-09-05 17:16:57 +00:00
target . AddNewMessages ( new InfoMessage ( "Supported commands: /help, /me [action], /join [channel], /chat [user], /np" ) ) ;
2018-04-14 11:32:48 +00:00
break ;
default :
2018-12-20 08:01:08 +00:00
target . AddNewMessages ( new ErrorMessage ( $@"""/{command}"" is not supported! For a list of supported commands see /help" ) ) ;
2018-04-14 11:32:48 +00:00
break ;
}
}
2022-10-28 07:22:35 +00:00
private void addMessages ( List < Message > messages )
2018-04-14 11:32:48 +00:00
{
var channels = JoinedChannels . ToList ( ) ;
2018-11-12 11:41:10 +00:00
foreach ( var group in messages . GroupBy ( m = > m . ChannelId ) )
2018-04-14 11:32:48 +00:00
channels . Find ( c = > c . Id = = group . Key ) ? . AddNewMessages ( group . ToArray ( ) ) ;
2022-11-02 08:13:14 +00:00
2022-11-12 14:32:05 +00:00
lastSilenceMessageId ? ? = messages . LastOrDefault ( ) ? . Id ;
2018-04-14 11:32:48 +00:00
}
2018-11-13 08:24:11 +00:00
private void initializeChannels ( )
2018-04-14 11:32:48 +00:00
{
2023-01-09 07:15:30 +00:00
// This request is self-retrying until it succeeds.
// To avoid requests piling up when not logged in (ie. API is unavailable) exit early.
2023-01-09 18:33:04 +00:00
if ( ! api . IsLoggedIn )
2023-01-09 07:15:30 +00:00
return ;
2018-04-14 11:32:48 +00:00
var req = new ListChannelsRequest ( ) ;
2021-10-27 04:04:41 +00:00
bool joinDefaults = JoinedChannels . Count = = 0 ;
2018-11-13 06:20:40 +00:00
2018-04-14 11:32:48 +00:00
req . Success + = channels = >
{
2018-07-24 15:51:20 +00:00
foreach ( var channel in channels )
{
2018-11-21 08:15:10 +00:00
var ch = getChannel ( channel , addToAvailable : true ) ;
2018-07-24 15:51:20 +00:00
// join any channels classified as "defaults"
2018-11-13 08:24:11 +00:00
if ( joinDefaults & & defaultChannels . Any ( c = > c . Equals ( channel . Name , StringComparison . OrdinalIgnoreCase ) ) )
2020-06-03 08:31:55 +00:00
joinChannel ( ch ) ;
2018-07-24 15:51:20 +00:00
}
2018-04-14 11:32:48 +00:00
} ;
2023-01-09 07:15:30 +00:00
2018-07-24 02:54:11 +00:00
req . Failure + = error = >
{
Logger . Error ( error , "Fetching channel list failed" ) ;
2023-01-09 07:15:30 +00:00
Scheduler . AddDelayed ( initializeChannels , 60000 ) ;
2018-07-24 02:54:11 +00:00
} ;
2018-04-14 11:32:48 +00:00
api . Queue ( req ) ;
}
2018-07-29 19:40:43 +00:00
/// <summary>
/// Fetches inital messages of a channel
///
/// TODO: remove this when the API supports returning initial fetch messages for more than one channel by specifying the last message id per channel instead of one last message id globally.
/// right now it caps out at 50 messages and therefore only returns one channel's worth of content.
/// </summary>
/// <param name="channel">The channel </param>
2021-12-10 05:04:31 +00:00
private void fetchInitialMessages ( Channel channel )
2018-07-29 19:40:43 +00:00
{
2020-06-03 08:31:55 +00:00
if ( channel . Id < = 0 | | channel . MessagesLoaded ) return ;
2018-11-13 08:24:11 +00:00
2018-11-12 11:41:10 +00:00
var fetchInitialMsgReq = new GetMessagesRequest ( channel ) ;
2018-11-13 08:24:11 +00:00
fetchInitialMsgReq . Success + = messages = >
{
2022-10-28 07:22:35 +00:00
addMessages ( messages ) ;
2018-11-14 04:19:20 +00:00
channel . MessagesLoaded = true ; // this will mark the channel as having received messages even if there were none.
2018-11-13 08:24:11 +00:00
} ;
2018-07-29 19:40:43 +00:00
api . Queue ( fetchInitialMsgReq ) ;
}
2022-11-02 08:13:14 +00:00
/// <summary>
/// Sends an acknowledgement request to the API.
/// This marks the user as online to receive messages from public channels, while also returning a list of silenced users.
2022-11-12 13:24:27 +00:00
/// It needs to be called at least once every 10 minutes to remain visibly marked as online.
2022-11-02 08:13:14 +00:00
/// </summary>
public void SendAck ( )
{
2022-11-12 14:02:37 +00:00
if ( apiState . Value ! = APIState . Online )
return ;
2022-11-02 08:13:14 +00:00
var req = new ChatAckRequest
{
2022-11-12 14:32:05 +00:00
SinceMessageId = lastSilenceMessageId ,
2022-11-02 08:13:14 +00:00
SinceSilenceId = lastSilenceId
} ;
2022-11-12 12:41:10 +00:00
req . Failure + = _ = > scheduleNextRequest ( ) ;
2022-11-02 08:13:14 +00:00
req . Success + = ack = >
{
foreach ( var silence in ack . Silences )
{
foreach ( var channel in JoinedChannels )
channel . RemoveMessagesFromUser ( silence . UserId ) ;
lastSilenceId = Math . Max ( lastSilenceId ? ? 0 , silence . Id ) ;
}
2022-11-12 12:41:10 +00:00
scheduleNextRequest ( ) ;
2022-11-02 08:13:14 +00:00
} ;
api . Queue ( req ) ;
2022-11-12 12:41:10 +00:00
void scheduleNextRequest ( )
{
scheduledAck ? . Cancel ( ) ;
2022-11-12 14:02:37 +00:00
scheduledAck = Scheduler . AddDelayed ( SendAck , 60000 ) ;
2022-11-12 12:41:10 +00:00
}
2022-11-02 08:13:14 +00:00
}
2018-11-21 08:15:10 +00:00
/// <summary>
/// Find an existing channel instance for the provided channel. Lookup is performed basd on ID.
/// The provided channel may be used if an existing instance is not found.
/// </summary>
/// <param name="lookup">A candidate channel to be used for lookup or permanently on lookup failure.</param>
/// <param name="addToAvailable">Whether the channel should be added to <see cref="AvailableChannels"/> if not already.</param>
/// <param name="addToJoined">Whether the channel should be added to <see cref="JoinedChannels"/> if not already.</param>
/// <returns>The found channel.</returns>
private Channel getChannel ( Channel lookup , bool addToAvailable = false , bool addToJoined = false )
2018-11-13 06:20:40 +00:00
{
2018-11-21 08:15:10 +00:00
Channel found = null ;
2018-11-13 06:20:40 +00:00
2022-11-07 03:25:23 +00:00
bool lookupCondition ( Channel ch )
{
if ( ch . Id > 0 & & lookup . Id > 0 )
return ch . Id = = lookup . Id ;
return ch . Name = = lookup . Name ;
}
2018-11-22 09:27:22 +00:00
var available = AvailableChannels . FirstOrDefault ( lookupCondition ) ;
2018-11-21 08:15:10 +00:00
if ( available ! = null )
found = available ;
2018-11-13 06:20:40 +00:00
2018-11-22 09:27:22 +00:00
var joined = JoinedChannels . FirstOrDefault ( lookupCondition ) ;
2018-11-21 08:15:10 +00:00
if ( found = = null & & joined ! = null )
found = joined ;
if ( found = = null )
2018-11-13 06:20:40 +00:00
{
2018-11-21 08:15:10 +00:00
found = lookup ;
// if we're using a channel object from the server, we want to remove ourselves from the users list.
// this is because we check the first user in the channel to display a name/icon on tabs for now.
var foundSelf = found . Users . FirstOrDefault ( u = > u . Id = = api . LocalUser . Value . Id ) ;
2018-11-13 06:20:40 +00:00
if ( foundSelf ! = null )
2018-11-21 08:15:10 +00:00
found . Users . Remove ( foundSelf ) ;
}
2022-11-07 05:34:53 +00:00
else
{
found . Id = lookup . Id ;
found . Name = lookup . Name ;
found . LastMessageId = Math . Max ( found . LastMessageId ? ? 0 , lookup . LastMessageId ? ? 0 ) ;
}
2018-11-21 08:15:10 +00:00
2018-11-21 22:21:27 +00:00
if ( joined = = null & & addToJoined ) joinedChannels . Add ( found ) ;
if ( available = = null & & addToAvailable ) availableChannels . Add ( found ) ;
2018-11-21 08:15:10 +00:00
return found ;
}
2018-11-13 06:20:40 +00:00
2018-11-21 08:15:10 +00:00
/// <summary>
2021-01-21 06:42:23 +00:00
/// Joins a channel if it has not already been joined. Must be called from the update thread.
2018-11-21 08:15:10 +00:00
/// </summary>
/// <param name="channel">The channel to join.</param>
/// <returns>The joined channel. Note that this may not match the parameter channel as it is a backed object.</returns>
2022-05-15 18:38:37 +00:00
public Channel JoinChannel ( Channel channel ) = > joinChannel ( channel , true ) ;
2020-06-03 08:31:55 +00:00
2022-05-15 18:38:37 +00:00
private Channel joinChannel ( Channel channel , bool fetchInitialMessages = false )
2018-11-21 08:15:10 +00:00
{
if ( channel = = null ) return null ;
2018-11-13 06:20:40 +00:00
2018-11-21 08:15:10 +00:00
channel = getChannel ( channel , addToJoined : true ) ;
// ensure we are joined to the channel
if ( ! channel . Joined . Value )
{
2020-06-08 08:49:45 +00:00
channel . Joined . Value = true ;
2020-06-03 08:31:55 +00:00
switch ( channel . Type )
2018-11-13 06:20:40 +00:00
{
2020-06-03 12:28:29 +00:00
case ChannelType . Multiplayer :
// join is implicit. happens when you join a multiplayer game.
// this will probably change in the future.
2022-05-15 18:38:37 +00:00
joinChannel ( channel , fetchInitialMessages ) ;
2020-06-03 12:28:29 +00:00
return channel ;
2020-06-08 08:49:45 +00:00
case ChannelType . PM :
2022-06-28 04:58:35 +00:00
Logger . Log ( $"Attempting to join PM channel {channel}" ) ;
2020-06-08 08:49:45 +00:00
var createRequest = new CreateChannelRequest ( channel ) ;
2022-06-28 04:58:35 +00:00
createRequest . Failure + = e = >
{
Logger . Log ( $"Failed to join PM channel {channel} ({e.Message})" ) ;
} ;
2020-06-08 08:49:45 +00:00
createRequest . Success + = resChannel = >
{
2022-06-28 04:58:35 +00:00
Logger . Log ( $"Joined PM channel {channel} ({resChannel.ChannelID})" ) ;
2020-06-08 08:49:45 +00:00
if ( resChannel . ChannelID . HasValue )
{
channel . Id = resChannel . ChannelID . Value ;
2022-10-28 07:22:35 +00:00
addMessages ( resChannel . RecentMessages ) ;
2020-06-08 08:49:45 +00:00
channel . MessagesLoaded = true ; // this will mark the channel as having received messages even if there were none.
}
} ;
api . Queue ( createRequest ) ;
2020-06-03 08:31:55 +00:00
break ;
default :
2022-06-28 04:58:35 +00:00
Logger . Log ( $"Attempting to join public channel {channel}" ) ;
2020-07-14 04:07:17 +00:00
var req = new JoinChannelRequest ( channel ) ;
2022-06-28 04:58:35 +00:00
req . Success + = ( ) = >
{
Logger . Log ( $"Joined public channel {channel}" ) ;
joinChannel ( channel , fetchInitialMessages ) ;
2022-12-07 05:23:52 +00:00
// Required after joining public channels to mark the user as online in them.
// Todo: Temporary workaround for https://github.com/ppy/osu-web/issues/9602
SendAck ( ) ;
2022-06-28 04:58:35 +00:00
} ;
req . Failure + = e = >
{
Logger . Log ( $"Failed to join public channel {channel} ({e.Message})" ) ;
LeaveChannel ( channel ) ;
} ;
2020-06-03 08:31:55 +00:00
api . Queue ( req ) ;
return channel ;
2018-11-13 06:20:40 +00:00
}
}
2020-06-03 08:31:55 +00:00
else
{
if ( fetchInitialMessages )
2021-12-10 05:04:31 +00:00
this . fetchInitialMessages ( channel ) ;
2020-06-03 08:31:55 +00:00
}
2018-11-13 06:20:40 +00:00
2022-05-15 18:38:37 +00:00
CurrentChannel . Value ? ? = channel ;
2018-11-13 06:20:40 +00:00
2018-11-21 08:15:10 +00:00
return channel ;
2018-11-13 06:20:40 +00:00
}
2021-01-21 06:42:23 +00:00
/// <summary>
/// Leave the specified channel. Can be called from any thread.
/// </summary>
/// <param name="channel">The channel to leave.</param>
2023-02-08 02:31:28 +00:00
public void LeaveChannel ( Channel channel ) = > Schedule ( ( ) = > leaveChannel ( channel , true ) ) ;
private void leaveChannel ( Channel channel , bool sendLeaveRequest )
2018-11-13 06:20:40 +00:00
{
if ( channel = = null ) return ;
2018-11-21 18:15:55 +00:00
if ( channel = = CurrentChannel . Value )
CurrentChannel . Value = null ;
2018-11-13 06:20:40 +00:00
2018-11-21 18:15:55 +00:00
joinedChannels . Remove ( channel ) ;
2020-12-14 01:46:02 +00:00
// Prevent the closedChannel list from exceeding the max size
// by removing the oldest element
2020-12-20 19:18:00 +00:00
if ( closedChannels . Count > = closed_channels_max_size )
2020-12-14 01:46:02 +00:00
{
2020-12-20 19:18:00 +00:00
closedChannels . RemoveAt ( 0 ) ;
2020-12-14 01:46:02 +00:00
}
2021-06-03 12:20:52 +00:00
// For PM channels, we store the user ID; else, we store the channel ID
2020-12-20 19:18:00 +00:00
closedChannels . Add ( channel . Type = = ChannelType . PM
? new ClosedChannel ( ChannelType . PM , channel . Users . Single ( ) . Id )
: new ClosedChannel ( channel . Type , channel . Id ) ) ;
2018-11-13 06:20:40 +00:00
if ( channel . Joined . Value )
{
2023-02-10 09:29:39 +00:00
if ( sendLeaveRequest )
api . Queue ( new LeaveChannelRequest ( channel ) ) ;
2018-11-13 06:20:40 +00:00
channel . Joined . Value = false ;
}
2023-02-08 02:31:28 +00:00
}
2018-11-13 06:20:40 +00:00
2020-12-14 01:46:02 +00:00
/// <summary>
2021-06-03 12:20:52 +00:00
/// Opens the most recently closed channel that has not already been reopened,
2020-12-14 02:02:35 +00:00
/// Works similarly to reopening the last closed tab on a web browser.
2020-12-14 01:46:02 +00:00
/// </summary>
2020-12-13 18:21:50 +00:00
public void JoinLastClosedChannel ( )
{
2021-06-03 12:20:52 +00:00
// This loop could be eliminated if the join channel operation ensured that every channel joined
// is removed from the closedChannels list, but it'd require a linear scan of closed channels on every join.
// To keep the overhead of joining channels low, just lazily scan the list of closed channels locally.
2020-12-20 19:18:00 +00:00
while ( closedChannels . Count > 0 )
2020-12-16 15:13:50 +00:00
{
2020-12-20 19:18:00 +00:00
ClosedChannel lastClosedChannel = closedChannels . Last ( ) ;
closedChannels . RemoveAt ( closedChannels . Count - 1 ) ;
2020-12-16 15:13:50 +00:00
2021-06-03 11:58:19 +00:00
// If the user has already joined the channel, try the next one
2021-06-03 12:13:01 +00:00
if ( joinedChannels . FirstOrDefault ( lastClosedChannel . Matches ) ! = null )
2021-06-03 11:58:19 +00:00
continue ;
2020-12-14 04:27:48 +00:00
2021-06-03 12:13:01 +00:00
Channel lastChannel = AvailableChannels . FirstOrDefault ( lastClosedChannel . Matches ) ;
2020-12-20 19:18:00 +00:00
2021-06-03 11:58:19 +00:00
if ( lastChannel ! = null )
{
2021-06-03 12:20:52 +00:00
// Channel exists as an available channel, directly join it
2021-06-03 11:58:19 +00:00
CurrentChannel . Value = JoinChannel ( lastChannel ) ;
}
else if ( lastClosedChannel . Type = = ChannelType . PM )
{
2021-06-03 12:20:52 +00:00
// Try to get user in order to open PM chat
2022-01-03 08:31:12 +00:00
users . GetUserAsync ( ( int ) lastClosedChannel . Id ) . ContinueWith ( task = >
2021-06-03 11:58:19 +00:00
{
2022-01-06 13:54:43 +00:00
var user = task . GetResultSafely ( ) ;
2022-01-05 06:54:10 +00:00
if ( user ! = null )
Schedule ( ( ) = > CurrentChannel . Value = JoinChannel ( new Channel ( user ) ) ) ;
2021-06-03 11:58:19 +00:00
} ) ;
2020-12-16 17:04:07 +00:00
}
2021-06-03 11:58:19 +00:00
return ;
2020-12-14 01:46:02 +00:00
}
2020-12-13 18:21:50 +00:00
}
2020-01-11 16:42:02 +00:00
/// <summary>
/// Marks the <paramref name="channel"/> as read
/// </summary>
/// <param name="channel">The channel that will be marked as read</param>
2020-01-11 18:47:35 +00:00
public void MarkChannelAsRead ( Channel channel )
2020-01-02 16:07:28 +00:00
{
2020-01-12 15:24:14 +00:00
if ( channel . LastMessageId = = channel . LastReadId )
return ;
2021-08-15 00:57:11 +00:00
var message = channel . Messages . FindLast ( msg = > ! ( msg is LocalMessage ) ) ;
2020-01-13 03:22:44 +00:00
if ( message = = null )
return ;
2020-01-02 16:07:28 +00:00
var req = new MarkChannelAsReadRequest ( channel , message ) ;
2020-01-11 16:42:02 +00:00
2020-01-02 16:07:28 +00:00
req . Success + = ( ) = > channel . LastReadId = message . Id ;
2022-02-24 07:02:16 +00:00
req . Failure + = e = > Logger . Log ( $"Failed to mark channel {channel} up to '{message}' as read ({e.Message})" , LoggingTarget . Network ) ;
2020-01-11 16:42:02 +00:00
2020-01-02 16:07:28 +00:00
api . Queue ( req ) ;
}
2022-10-28 09:08:08 +00:00
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
connector ? . Dispose ( ) ;
}
2018-04-14 11:32:48 +00:00
}
2018-07-09 16:45:11 +00:00
2018-07-24 03:14:33 +00:00
/// <summary>
/// An exception thrown when a channel could not been found.
/// </summary>
2018-07-09 16:45:11 +00:00
public class ChannelNotFoundException : Exception
{
public ChannelNotFoundException ( string channelName )
: base ( $"A channel with the name {channelName} could not be found." )
{
}
}
2020-12-20 19:18:00 +00:00
/// <summary>
2021-06-03 12:20:52 +00:00
/// Stores information about a closed channel
2020-12-20 19:18:00 +00:00
/// </summary>
public class ClosedChannel
{
2021-06-03 05:56:21 +00:00
public readonly ChannelType Type ;
public readonly long Id ;
2020-12-20 19:18:00 +00:00
public ClosedChannel ( ChannelType type , long id )
{
Type = type ;
Id = id ;
}
2021-06-03 12:13:01 +00:00
public bool Matches ( Channel channel )
2020-12-20 19:18:00 +00:00
{
2020-12-20 19:51:39 +00:00
if ( channel . Type ! = Type ) return false ;
2020-12-20 19:18:00 +00:00
2020-12-20 19:51:39 +00:00
return Type = = ChannelType . PM
? channel . Users . Single ( ) . Id = = Id
: channel . Id = = Id ;
2020-12-20 19:18:00 +00:00
}
}
2018-04-14 11:32:48 +00:00
}