2024-08-04 23:46:10 +00:00
|
|
|
using Serilog;
|
|
|
|
using TwitchChatTTS.Twitch.Socket.Messages;
|
|
|
|
|
|
|
|
namespace TwitchChatTTS.Twitch.Socket.Handlers
|
|
|
|
{
|
|
|
|
public class SessionWelcomeHandler : ITwitchSocketHandler
|
|
|
|
{
|
|
|
|
public string Name => "session_welcome";
|
|
|
|
|
|
|
|
private readonly TwitchApiClient _api;
|
|
|
|
private readonly User _user;
|
|
|
|
private readonly ILogger _logger;
|
|
|
|
|
2024-08-11 21:22:37 +00:00
|
|
|
public SessionWelcomeHandler(TwitchApiClient api, User user, ILogger logger)
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
|
|
|
_api = api;
|
|
|
|
_user = user;
|
|
|
|
_logger = logger;
|
|
|
|
}
|
|
|
|
|
2024-08-06 19:29:29 +00:00
|
|
|
public async Task Execute(TwitchWebsocketClient sender, object data)
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
|
|
|
if (data is not SessionWelcomeMessage message)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (string.IsNullOrEmpty(message.Session.Id))
|
|
|
|
{
|
|
|
|
_logger.Warning($"No session info provided by Twitch [status: {message.Session.Status}]");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-07 22:11:09 +00:00
|
|
|
int waited = 0;
|
2024-08-11 21:22:37 +00:00
|
|
|
while ((_user.TwitchUserId <= 0 || _user.TwitchConnection == null) && ++waited < 5)
|
2024-08-07 22:11:09 +00:00
|
|
|
await Task.Delay(TimeSpan.FromSeconds(1));
|
|
|
|
|
2024-08-11 21:22:37 +00:00
|
|
|
if (_user.TwitchConnection == null)
|
2024-08-07 22:01:04 +00:00
|
|
|
{
|
2024-08-11 21:22:37 +00:00
|
|
|
_logger.Error("Ensure you have linked either your Twitch account or TTS' bot to your TTS account. Twitch client will not be connecting.");
|
|
|
|
return;
|
2024-08-07 22:01:04 +00:00
|
|
|
}
|
2024-08-11 21:22:37 +00:00
|
|
|
|
|
|
|
_api.Initialize(_user.TwitchConnection.ClientId, _user.TwitchConnection.AccessToken);
|
2024-08-12 07:54:38 +00:00
|
|
|
var span = _user.TwitchConnection.ExpiresAt - DateTime.Now;
|
|
|
|
var timeLeft = span.Days >= 2 ? span.Days + " days" : (span.Hours >= 2 ? span.Hours + " hours" : span.Minutes + " minutes");
|
|
|
|
if (span.Days >= 3)
|
2024-08-11 21:22:37 +00:00
|
|
|
_logger.Information($"Twitch connection has {timeLeft} before it is revoked.");
|
2024-08-12 07:54:38 +00:00
|
|
|
else if (span.Minutes >= 0)
|
2024-08-11 21:22:37 +00:00
|
|
|
_logger.Warning($"Twitch connection has {timeLeft} before it is revoked. Refreshing the token is soon required.");
|
2024-08-12 18:06:10 +00:00
|
|
|
else
|
|
|
|
{
|
2024-08-11 21:22:37 +00:00
|
|
|
_logger.Error("Twitch connection has its permissions revoked. Refresh the token. Twith client will not be connecting.");
|
2024-08-07 22:01:04 +00:00
|
|
|
return;
|
|
|
|
}
|
2024-08-06 19:29:29 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
if (!sender.TwitchReconnected)
|
2024-08-06 19:29:29 +00:00
|
|
|
{
|
2024-08-12 18:06:10 +00:00
|
|
|
string broadcasterId = _user.TwitchUserId.ToString();
|
|
|
|
string[] subscriptionsv1 = [
|
|
|
|
"channel.chat.message",
|
|
|
|
"channel.chat.message_delete",
|
|
|
|
"channel.chat.clear",
|
|
|
|
"channel.chat.clear_user_messages",
|
|
|
|
"channel.subscribe",
|
|
|
|
"channel.subscription.gift",
|
|
|
|
"channel.subscription.message",
|
|
|
|
"channel.ad_break.begin",
|
|
|
|
"channel.ban",
|
|
|
|
"channel.channel_points_custom_reward_redemption.add"
|
|
|
|
];
|
|
|
|
string[] subscriptionsv2 = [
|
|
|
|
"channel.follow",
|
|
|
|
];
|
|
|
|
|
|
|
|
string? pagination = null;
|
|
|
|
int size = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
var subscriptionsData = await _api.GetSubscriptions(status: "enabled", broadcasterId: broadcasterId, after: pagination);
|
|
|
|
var subscriptionNames = subscriptionsData?.Data == null ? [] : subscriptionsData.Data.Select(s => s.Type).ToArray();
|
2024-08-06 19:29:29 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
if (subscriptionNames.Length == 0)
|
|
|
|
break;
|
2024-08-06 19:29:29 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
foreach (var d in subscriptionsData!.Data!)
|
|
|
|
sender.AddSubscription(broadcasterId, d.Type, d.Id);
|
2024-08-06 19:29:29 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
subscriptionsv1 = subscriptionsv1.Except(subscriptionNames).ToArray();
|
|
|
|
subscriptionsv2 = subscriptionsv2.Except(subscriptionNames).ToArray();
|
2024-08-06 19:29:29 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
pagination = subscriptionsData?.Pagination?.Cursor;
|
|
|
|
size = subscriptionNames.Length;
|
|
|
|
} while (size >= 100 && pagination != null && subscriptionsv1.Length + subscriptionsv2.Length > 0);
|
2024-08-06 19:29:29 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
foreach (var subscription in subscriptionsv1)
|
|
|
|
await Subscribe(sender, subscription, message.Session.Id, broadcasterId, "1");
|
|
|
|
foreach (var subscription in subscriptionsv2)
|
|
|
|
await Subscribe(sender, subscription, message.Session.Id, broadcasterId, "2");
|
2024-08-11 21:22:37 +00:00
|
|
|
|
2024-08-12 18:06:10 +00:00
|
|
|
await Subscribe(sender, "channel.raid", broadcasterId, async () => await _api.CreateChannelRaidEventSubscription("1", message.Session.Id, to: broadcasterId));
|
|
|
|
}
|
2024-08-06 19:29:29 +00:00
|
|
|
|
|
|
|
sender.Identify(message.Session.Id);
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
|
2024-08-06 19:29:29 +00:00
|
|
|
private async Task Subscribe(TwitchWebsocketClient sender, string subscriptionName, string sessionId, string broadcasterId, string version)
|
2024-08-04 23:46:10 +00:00
|
|
|
{
|
2024-08-12 18:06:10 +00:00
|
|
|
await Subscribe(sender, subscriptionName, broadcasterId, async () => await _api.CreateEventSubscription(subscriptionName, version, sessionId, broadcasterId));
|
2024-08-07 22:01:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private async Task Subscribe(TwitchWebsocketClient sender, string subscriptionName, string broadcasterId, Func<Task<EventResponse<NotificationInfo>?>> subscribe)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
var response = await subscribe();
|
|
|
|
if (response == null)
|
|
|
|
{
|
|
|
|
return;
|
2024-08-04 23:46:10 +00:00
|
|
|
}
|
|
|
|
if (response.Data == null)
|
|
|
|
{
|
|
|
|
_logger.Error($"Failed to create an event subscription [subscription type: {subscriptionName}][reason: data is null]");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!response.Data.Any())
|
|
|
|
{
|
|
|
|
_logger.Error($"Failed to create an event subscription [subscription type: {subscriptionName}][reason: data is empty]");
|
|
|
|
return;
|
|
|
|
}
|
2024-08-06 19:29:29 +00:00
|
|
|
|
|
|
|
foreach (var d in response.Data)
|
|
|
|
sender.AddSubscription(broadcasterId, d.Type, d.Id);
|
|
|
|
|
2024-08-04 23:46:10 +00:00
|
|
|
_logger.Information($"Sucessfully added subscription to Twitch websockets [subscription type: {subscriptionName}]");
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.Error(ex, $"Failed to create an event subscription [subscription type: {subscriptionName}][reason: exception]");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|