2024-08-06 19:29:29 +00:00
using CommonSocketLibrary.Abstract ;
using Microsoft.Extensions.DependencyInjection ;
using Serilog ;
using TwitchChatTTS.Twitch.Socket.Messages ;
namespace TwitchChatTTS.Twitch.Socket
{
public interface ITwitchConnectionManager
{
TwitchWebsocketClient GetWorkingClient ( ) ;
TwitchWebsocketClient GetBackupClient ( ) ;
}
public class TwitchConnectionManager : ITwitchConnectionManager
{
private TwitchWebsocketClient ? _identified ;
private TwitchWebsocketClient ? _backup ;
private readonly IServiceProvider _serviceProvider ;
private readonly ILogger _logger ;
private readonly object _lock ;
public TwitchConnectionManager ( IServiceProvider serviceProvider , ILogger logger )
{
_serviceProvider = serviceProvider ;
_logger = logger ;
_lock = new object ( ) ;
}
public TwitchWebsocketClient GetBackupClient ( )
{
lock ( _lock )
{
if ( _identified = = null )
throw new InvalidOperationException ( "Cannot get backup Twitch client yet. Waiting for identification." ) ;
if ( _backup ! = null )
return _backup ;
return CreateNewClient ( ) ;
}
}
public TwitchWebsocketClient GetWorkingClient ( )
{
lock ( _lock )
{
if ( _identified = = null )
{
return CreateNewClient ( ) ;
}
return _identified ;
}
}
private TwitchWebsocketClient CreateNewClient ( )
{
if ( _backup ! = null )
return _backup ;
var client = ( _serviceProvider . GetRequiredKeyedService < SocketClient < TwitchWebsocketMessage > > ( "twitch-create" ) as TwitchWebsocketClient ) ! ;
client . Initialize ( ) ;
_backup = client ;
2024-08-07 22:01:04 +00:00
client . OnIdentified + = async ( _ , _ ) = > await OnIdentified ( client ) ;
client . OnDisconnected + = async ( _ , _ ) = > await OnDisconnection ( client ) ;
_logger . Debug ( "Created a Twitch websocket client." ) ;
return client ;
}
private async Task OnDisconnection ( TwitchWebsocketClient client )
{
bool reconnecting = false ;
lock ( _lock )
2024-08-06 19:29:29 +00:00
{
2024-08-07 22:01:04 +00:00
if ( _identified ? . UID = = client . UID )
2024-08-06 19:29:29 +00:00
{
2024-08-14 18:19:18 +00:00
_logger . Debug ( $"Identified Twitch client has disconnected [client: {client.UID}][main: {_identified.UID}][backup: {_backup?.UID}]" ) ;
2024-08-07 22:01:04 +00:00
_identified = null ;
reconnecting = true ;
}
else if ( _backup ? . UID = = client . UID )
{
2024-08-14 18:19:18 +00:00
_logger . Debug ( $"Backup Twitch client has disconnected [client: {client.UID}][main: {_identified?.UID}][backup: {_backup.UID}]" ) ;
2024-08-07 22:01:04 +00:00
_backup = null ;
}
else if ( client . ReceivedReconnecting )
_logger . Debug ( $"Twitch client disconnected due to reconnection [client: {client.UID}][main: {_identified?.UID}][backup: {_backup?.UID}]" ) ;
else
2024-08-14 18:19:18 +00:00
_logger . Warning ( $"Twitch client disconnected from unknown source [client: {client.UID}][main: {_identified?.UID}][backup: {_backup?.UID}]" ) ;
2024-08-07 22:01:04 +00:00
}
2024-08-06 19:29:29 +00:00
2024-08-07 22:01:04 +00:00
if ( reconnecting )
2024-08-06 19:29:29 +00:00
{
2024-08-07 22:01:04 +00:00
var newClient = GetWorkingClient ( ) ;
2024-08-12 17:55:28 +00:00
await newClient . Reconnect ( ) ;
2024-08-07 22:01:04 +00:00
}
}
private async Task OnIdentified ( TwitchWebsocketClient client )
{
bool clientDisconnect = false ;
lock ( _lock )
{
if ( _identified = = null | | _identified . ReceivedReconnecting )
2024-08-06 19:29:29 +00:00
{
2024-08-07 22:01:04 +00:00
if ( _backup ! = null & & _backup . UID = = client . UID )
2024-08-06 19:29:29 +00:00
{
2024-08-14 18:19:18 +00:00
_logger . Debug ( $"Twitch client has been identified [client: {client.UID}][main: {_identified?.UID}][backup: {_backup.UID}]" ) ;
2024-08-07 22:01:04 +00:00
_identified = _backup ;
2024-08-06 19:29:29 +00:00
_backup = null ;
}
else
2024-08-07 22:01:04 +00:00
_logger . Warning ( $"Twitch client identified from unknown sources [client: {client.UID}][main: {_identified?.UID}][backup: {_backup?.UID}]" ) ;
2024-08-06 19:29:29 +00:00
}
2024-08-07 22:01:04 +00:00
else if ( _identified . UID = = client . UID )
{
2024-08-14 18:19:18 +00:00
_logger . Debug ( $"Twitch client has been re-identified [client: {client.UID}][main: {_identified.UID}][backup: {_backup?.UID}]" ) ;
2024-08-07 22:01:04 +00:00
}
else if ( _backup = = null | | _backup . UID ! = client . UID )
2024-08-07 19:32:44 +00:00
{
2024-08-07 22:01:04 +00:00
_logger . Warning ( $"Twitch client has been identified, but isn't main or backup [client: {client.UID}][main: {_identified.UID}][backup: {_backup?.UID}]" ) ;
clientDisconnect = true ;
2024-08-07 19:32:44 +00:00
}
2024-08-07 22:01:04 +00:00
}
2024-08-06 19:29:29 +00:00
2024-08-07 22:01:04 +00:00
if ( clientDisconnect )
await client . DisconnectAsync ( new SocketDisconnectionEventArgs ( "Closed" , "No need for a tertiary client." ) ) ;
2024-08-06 19:29:29 +00:00
}
}
}