2024-08-04 23:46:10 +00:00
|
|
|
using CommonSocketLibrary.Abstract;
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
|
|
using Serilog;
|
|
|
|
using System.Text.Json;
|
|
|
|
using System.Net.WebSockets;
|
|
|
|
using TwitchChatTTS.Twitch.Socket.Messages;
|
|
|
|
using System.Text;
|
|
|
|
using TwitchChatTTS.Twitch.Socket.Handlers;
|
2024-08-06 19:29:29 +00:00
|
|
|
using CommonSocketLibrary.Backoff;
|
2024-08-04 23:46:10 +00:00
|
|
|
|
|
|
|
namespace TwitchChatTTS.Twitch.Socket
|
|
|
|
{
|
|
|
|
public class TwitchWebsocketClient : SocketClient<TwitchWebsocketMessage>
|
|
|
|
{
|
2024-08-06 19:29:29 +00:00
|
|
|
private readonly IDictionary<string, ITwitchSocketHandler> _handlers;
|
|
|
|
private readonly IDictionary<string, Type> _messageTypes;
|
|
|
|
private readonly IDictionary<string, string> _subscriptions;
|
|
|
|
private readonly IBackoff _backoff;
|
2024-08-07 19:32:44 +00:00
|
|
|
private readonly Configuration _configuration;
|
2024-08-06 19:29:29 +00:00
|
|
|
private bool _disconnected;
|
|
|
|
private readonly object _lock;
|
2024-08-04 23:46:10 +00:00
|
|
|
|
2024-08-06 19:29:29 +00:00
|
|
|
public event EventHandler<EventArgs> OnIdentified;
|
2024-08-04 23:46:10 +00:00
|
|
|
|
2024-08-06 21:15:05 +00:00
|
|
|
public string UID { get; }
|
2024-08-06 19:29:29 +00:00
|
|
|
public string URL;
|
|
|
|
public bool Connected { get; private set; }
|
|
|
|
public bool Identified { get; private set; }
|
|
|
|
public string SessionId { get; private set; }
|
|
|
|
public bool ReceivedReconnecting { get; set; }
|
2024-08-12 18:06:10 +00:00
|
|
|
public bool TwitchReconnected { get; set; }
|
2024-08-04 23:46:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
public TwitchWebsocketClient(
|
|
|
|
[FromKeyedServices("twitch")] IEnumerable<ITwitchSocketHandler> handlers,
|
2024-08-06 19:29:29 +00:00
|
|
|
[FromKeyedServices("twitch")] IBackoff backoff,
|
2024-08-07 19:32:44 +00:00
|
|
|
Configuration configuration,
|
2024-08-04 23:46:10 +00:00
|
|
|
ILogger logger
|
|
|
|
) : base(logger, new JsonSerializerOptions()
|
|
|
|
{
|
|
|
|
PropertyNameCaseInsensitive = false,
|
|
|
|
PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower
|
|
|
|
})
|
|
|
|
{
|
|
|
|
_handlers = handlers.ToDictionary(h => h.Name, h => h);
|
2024-08-06 19:29:29 +00:00
|
|
|
_backoff = backoff;
|
2024-08-07 19:32:44 +00:00
|
|
|
_configuration = configuration;
|
2024-08-06 19:29:29 +00:00
|
|
|
_subscriptions = new Dictionary<string, string>();
|
|
|
|
_lock = new object();
|
2024-08-04 23:46:10 +00:00
|
|
|
|
|
|
|
_messageTypes = new Dictionary<string, Type>();
|
2024-08-06 19:29:29 +00:00
|
|
|
_messageTypes.Add("session_keepalive", typeof(object));
|
2024-08-04 23:46:10 +00:00
|
|
|
_messageTypes.Add("session_welcome", typeof(SessionWelcomeMessage));
|
|
|
|
_messageTypes.Add("session_reconnect", typeof(SessionWelcomeMessage));
|
|
|
|
_messageTypes.Add("notification", typeof(NotificationMessage));
|
|
|
|
|
2024-08-06 21:15:05 +00:00
|
|
|
UID = Guid.NewGuid().ToString("D");
|
2024-08-12 07:54:38 +00:00
|
|
|
|
2024-08-07 19:32:44 +00:00
|
|
|
if (_configuration.Environment == "PROD" || string.IsNullOrWhiteSpace(_configuration.Twitch?.WebsocketUrl))
|
|
|
|
URL = "wss://eventsub.wss.twitch.tv/ws";
|
|
|
|
else
|
|
|
|
URL = _configuration.Twitch.WebsocketUrl;
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-08-06 19:29:29 +00:00
|
|
|
public void AddSubscription(string broadcasterId, string type, string id)
|
|
|
|
{
|
|
|
|
if (_subscriptions.ContainsKey(broadcasterId + '|' + type))
|
|
|
|
_subscriptions[broadcasterId + '|' + type] = id;
|
|
|
|
else
|
|
|
|
_subscriptions.Add(broadcasterId + '|' + type, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
public string? GetSubscriptionId(string broadcasterId, string type)
|
|
|
|
{
|
|
|
|
if (_subscriptions.TryGetValue(broadcasterId + '|' + type, out var id))
|
|
|
|
return id;
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSubscription(string broadcasterId, string type)
|
|
|
|
{
|
|
|
|
_subscriptions.Remove(broadcasterId + '|' + type);
|
|
|
|
}
|
|
|
|
|
2024-08-04 23:46:10 +00:00
|
|
|
public void Initialize()
|
|
|
|
{
|
2024-08-06 19:29:29 +00:00
|
|
|
_logger.Information($"Initializing Twitch websocket client.");
|
2024-08-04 23:46:10 +00:00
|
|
|
OnConnected += (sender, e) =>
|
|
|
|
{
|
|
|
|
Connected = true;
|
|
|
|
_logger.Information("Twitch websocket client connected.");
|
2024-08-06 19:29:29 +00:00
|
|
|
_disconnected = false;
|
2024-08-04 23:46:10 +00:00
|
|
|
};
|
|
|
|
|
2024-12-28 21:19:28 +00:00
|
|
|
OnDisconnected += (sender, e) =>
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
2024-08-06 19:29:29 +00:00
|
|
|
lock (_lock)
|
|
|
|
{
|
|
|
|
if (_disconnected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
_disconnected = true;
|
|
|
|
}
|
|
|
|
|
2024-08-07 19:32:44 +00:00
|
|
|
_logger.Information($"Twitch websocket client disconnected [status: {e.Status}][reason: {e.Reason}][client: {UID}]");
|
2024-08-04 23:46:10 +00:00
|
|
|
|
|
|
|
Connected = false;
|
|
|
|
Identified = false;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-08-12 20:10:11 +00:00
|
|
|
public override async Task Connect()
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
|
|
|
if (string.IsNullOrWhiteSpace(URL))
|
|
|
|
{
|
|
|
|
_logger.Warning("Lacking connection info for Twitch websockets. Not connecting to Twitch.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_logger.Debug($"Twitch websocket client attempting to connect to {URL}");
|
2024-08-06 19:29:29 +00:00
|
|
|
await ConnectAsync(URL);
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
|
2024-08-12 20:10:11 +00:00
|
|
|
public async Task Reconnect() => await Reconnect(_backoff);
|
2024-08-12 17:55:28 +00:00
|
|
|
|
2024-08-06 19:29:29 +00:00
|
|
|
public void Identify(string sessionId)
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
2024-08-06 19:29:29 +00:00
|
|
|
Identified = true;
|
|
|
|
SessionId = sessionId;
|
|
|
|
OnIdentified?.Invoke(this, EventArgs.Empty);
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
protected TwitchWebsocketMessage GenerateMessage<T>(string messageType, T data)
|
|
|
|
{
|
|
|
|
var metadata = new TwitchMessageMetadata()
|
|
|
|
{
|
|
|
|
MessageId = Guid.NewGuid().ToString(),
|
|
|
|
MessageType = messageType,
|
|
|
|
MessageTimestamp = DateTime.UtcNow
|
|
|
|
};
|
|
|
|
return new TwitchWebsocketMessage()
|
|
|
|
{
|
|
|
|
Metadata = metadata,
|
|
|
|
Payload = data
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-08-12 07:54:38 +00:00
|
|
|
protected override Task OnResponseReceived(TwitchWebsocketMessage? message)
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
2024-08-12 07:54:38 +00:00
|
|
|
return Task.Run(async () =>
|
2024-08-06 19:29:29 +00:00
|
|
|
{
|
2024-08-12 07:54:38 +00:00
|
|
|
if (message == null || message.Metadata == null)
|
|
|
|
{
|
|
|
|
_logger.Information("Twitch message is null");
|
|
|
|
return;
|
|
|
|
}
|
2024-08-04 23:46:10 +00:00
|
|
|
|
2024-08-12 07:54:38 +00:00
|
|
|
string content = message.Payload?.ToString() ?? string.Empty;
|
|
|
|
if (message.Metadata.MessageType != "session_keepalive")
|
|
|
|
_logger.Debug("Twitch RX #" + message.Metadata.MessageType + ": " + content);
|
2024-08-04 23:46:10 +00:00
|
|
|
|
2024-08-12 07:54:38 +00:00
|
|
|
if (!_messageTypes.TryGetValue(message.Metadata.MessageType, out var type) || type == null)
|
|
|
|
{
|
|
|
|
_logger.Debug($"Could not find Twitch message type [message type: {message.Metadata.MessageType}]");
|
|
|
|
return;
|
|
|
|
}
|
2024-08-04 23:46:10 +00:00
|
|
|
|
2024-08-12 07:54:38 +00:00
|
|
|
if (!_handlers.TryGetValue(message.Metadata.MessageType, out ITwitchSocketHandler? handler) || handler == null)
|
|
|
|
{
|
|
|
|
_logger.Debug($"Could not find Twitch handler [message type: {message.Metadata.MessageType}]");
|
|
|
|
return;
|
|
|
|
}
|
2024-08-04 23:46:10 +00:00
|
|
|
|
2024-08-12 07:54:38 +00:00
|
|
|
var data = JsonSerializer.Deserialize(content, type, _options);
|
|
|
|
if (data == null)
|
|
|
|
{
|
|
|
|
_logger.Warning("Twitch websocket message payload is null.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
await handler.Execute(this, data);
|
|
|
|
});
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public async Task Send<T>(string type, T data)
|
|
|
|
{
|
|
|
|
if (_socket == null || type == null || data == null)
|
|
|
|
return;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
var message = GenerateMessage(type, data);
|
|
|
|
var content = JsonSerializer.Serialize(message, _options);
|
|
|
|
|
|
|
|
var bytes = Encoding.UTF8.GetBytes(content);
|
|
|
|
var array = new ArraySegment<byte>(bytes);
|
|
|
|
var total = bytes.Length;
|
|
|
|
var current = 0;
|
|
|
|
|
|
|
|
while (current < total)
|
|
|
|
{
|
|
|
|
var size = Encoding.UTF8.GetBytes(content.Substring(current), array);
|
|
|
|
await _socket!.SendAsync(array, WebSocketMessageType.Text, current + size >= total, _cts!.Token);
|
|
|
|
current += size;
|
|
|
|
}
|
2024-08-06 19:29:29 +00:00
|
|
|
_logger.Debug("Twitch TX #" + type + ": " + content);
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
|
|
|
if (_socket.State.ToString().Contains("Close") || _socket.State == WebSocketState.Aborted)
|
|
|
|
{
|
|
|
|
await DisconnectAsync(new SocketDisconnectionEventArgs(_socket.CloseStatus.ToString()!, _socket.CloseStatusDescription ?? string.Empty));
|
|
|
|
_logger.Warning($"Socket state on closing = {_socket.State} | {_socket.CloseStatus?.ToString()} | {_socket.CloseStatusDescription}");
|
|
|
|
}
|
|
|
|
_logger.Error(e, $"Failed to send a websocket message [message type: {type}]");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|