using KfChatDotNetKickBot.Services; using KfChatDotNetWsClient; using KfChatDotNetWsClient.Models; using KfChatDotNetWsClient.Models.Events; using KfChatDotNetWsClient.Models.Json; using KickWsClient.Models; using Newtonsoft.Json; using NLog; using Spectre.Console; using Websocket.Client; namespace KfChatDotNetKickBot; public class KickBot { private ChatClient _kfClient; private KickWsClient.KickWsClient _kickClient; private Logger _logger = LogManager.GetCurrentClassLogger(); private Models.ConfigModel _config; private Thread _pingThread; private bool _pingEnabled = true; private bool _gambaSeshPresent = false; private string _xfSessionToken; // Oh no it's an ever expanding list that may never get cleaned up! // BUY MORE RAM private List _seenMsgIds = new List(); // Suppresses the command handler on initial start so it doesn't pick up things already handled on restart private bool _initialStartCooldown = true; public KickBot() { _logger.Info("Bot starting!"); const string configPath = "config.json"; if (!Path.Exists(configPath)) { _logger.Error($"{configPath} is missing! Exiting"); Environment.Exit(1); } _config = JsonConvert.DeserializeObject(File.ReadAllText(configPath)) ?? throw new InvalidOperationException(); RefreshXfToken(); _kfClient = new ChatClient(new ChatClientConfigModel { WsUri = _config.KfWsEndpoint, XfSessionToken = _xfSessionToken, CookieDomain = _config.KfWsEndpoint.Host, Proxy = _config.KfProxy, ReconnectTimeout = _config.KfReconnectTimeout }); _kickClient = new KickWsClient.KickWsClient(_config.PusherEndpoint.ToString(), _config.PusherProxy, _config.PusherReconnectTimeout); _kfClient.OnMessages += OnKfChatMessage; _kfClient.OnUsersParted += OnUsersParted; _kfClient.OnUsersJoined += OnUsersJoined; _kfClient.OnWsDisconnection += OnKfWsDisconnected; _kfClient.OnWsReconnect += OnKfWsReconnected; _kfClient.OnFailedToJoinRoom += OnFailedToJoinRoom; _kickClient.OnStreamerIsLive += OnStreamerIsLive; _kickClient.OnChatMessage += OnKickChatMessage; _kickClient.OnWsReconnect += OnPusherWsReconnected; _kickClient.OnPusherSubscriptionSucceeded += OnPusherSubscriptionSucceeded; _kickClient.OnStopStreamBroadcast += OnStopStreamBroadcast; _kfClient.StartWsClient().Wait(); _kfClient.JoinRoom(_config.KfChatRoomId); _kickClient.StartWsClient().Wait(); foreach (var channel in _config.PusherChannels) { _kickClient.SendPusherSubscribe(channel); } _pingThread = new Thread(PingThread); _pingThread.Start(); while (true) { var input = AnsiConsole.Prompt(new TextPrompt("Enter Message:")); _kfClient.SendMessage(input); } } private void OnFailedToJoinRoom(object sender, string message) { _logger.Error($"Couldn't join the room. KF returned: {message}"); _logger.Error("This is likely due to the session cookie expiring. Retrieving a new one."); RefreshXfToken(); _kfClient.UpdateToken(_xfSessionToken); _logger.Info("Retrieved fresh token. Reconnecting."); _kfClient.Disconnect(); _kfClient.StartWsClient().Wait(); _logger.Info("Client should be reconnecting now"); } private void PingThread() { while (_pingEnabled) { Thread.Sleep(TimeSpan.FromSeconds(15)); _logger.Debug("Pinging KF and Pusher"); _kfClient.SendMessage("/ping"); _kickClient.SendPusherPing(); if (_initialStartCooldown) _initialStartCooldown = false; } } private void RefreshXfToken() { var cookie = KfTokenService.FetchSessionTokenAsync(_config.KfDomain, _config.KfUsername, _config.KfPassword, _config.ChromiumPath, _config.KfProxy).Result; _logger.Debug($"FetchSessionTokenAsync returned {cookie}"); _xfSessionToken = cookie; } private void OnStreamerIsLive(object sender, KickModels.StreamerIsLiveEventModel? e) { AnsiConsole.MarkupLine("[green]Streamer is live!!![/]"); _sendChatMessage($"Bossman Live! {e?.Livestream.SessionTitle} https://kick.com/bossmanjack [i]This action was automated"); } private void OnStopStreamBroadcast(object sender, KickModels.StopStreamBroadcastEventModel? e) { AnsiConsole.MarkupLine("[green]Stream stopped!!![/]"); _sendChatMessage("The stream is so over. [i]This action was automated"); } private void OnKfChatMessage(object sender, List messages, MessagesJsonModel jsonPayload) { _logger.Debug($"Received {messages.Count} message(s)"); foreach (var message in messages) { AnsiConsole.MarkupLine($"[yellow]KF[/] <{message.Author.Username}> {message.Message.EscapeMarkup()} ({message.MessageDate.LocalDateTime.ToShortTimeString()})"); if (!_seenMsgIds.Contains(message.MessageId) && !_initialStartCooldown) { if (message.MessageRaw.StartsWith("!time")) { var bmt = new DateTimeOffset(TimeZoneInfo.ConvertTimeFromUtc(DateTime.UtcNow, TimeZoneInfo.FindSystemTimeZoneById("Eastern Standard Time")), TimeZoneInfo.FindSystemTimeZoneById("Eastern Standard Time").BaseUtcOffset); _sendChatMessage($"It's currently {bmt:h:mm:ss tt} BMT"); } else if (message.MessageRaw.StartsWith("!twisted")) { _sendChatMessage("🦍 🗣 GET IT TWISTED 🌪 , GAMBLE ✅ . PLEASE START GAMBLING 👍 . GAMBLING IS AN INVESTMENT 🎰 AND AN INVESTMENT ONLY 👍 . YOU WILL PROFIT 💰 , YOU WILL WIN ❗ ️. YOU WILL DO ALL OF THAT 💯 , YOU UNDERSTAND ⁉ ️ YOU WILL BECOME A BILLIONAIRE 💵 📈 AND REBUILD YOUR FUCKING LIFE 🤯"); } else if (message.MessageRaw.StartsWith("!insanity")) { _sendChatMessage("definition of insanity = doing the same thing over and over and over excecting a different result, and heres my dumbass trying to get rich every day and losing everythign i fucking touch every fucking time FUCK this bullshit FUCK MY LIEFdefinition of insanity = doing the same thing over and over and over excecting a different result, and heres my dumbass trying to get rich every day and losing everythign i fucking touch every fucking time FUCK this bullshit FUCK MY LIEF"); } else if (message.MessageRaw.StartsWith("!help")) { _sendChatMessage("[img]https://i.postimg.cc/fTw6tGWZ/ineedmoneydumbfuck.png[/img]", true); } } else { AnsiConsole.MarkupLine($"[blue]_seenMsgIds check => {!_seenMsgIds.Contains(message.MessageId)}, _initialStartCooldown => {_initialStartCooldown}[/]"); } _seenMsgIds.Add(message.MessageId); } } private void _sendChatMessage(string message, bool bypassSeshDetect = false) { if (_gambaSeshPresent && _config.EnableGambaSeshDetect && !bypassSeshDetect) { AnsiConsole.MarkupLine($"[red]Not sending message '{message.EscapeMarkup()}' as GambaSesh is present[/]"); return; } _kfClient.SendMessage(message); } private void OnKickChatMessage(object sender, KickModels.ChatMessageEventModel? e) { if (e == null) return; AnsiConsole.MarkupLine($"[green]Kick[/] <{e.Sender.Username}> {e.Content.EscapeMarkup()} ({e.CreatedAt.LocalDateTime.ToShortTimeString()})"); AnsiConsole.MarkupLine($"[cyan]BB Code Translation: {e.Content.TranslateKickEmotes().EscapeMarkup()}[/]"); if (e.Sender.Slug == "bossmanjack") { AnsiConsole.MarkupLine("[green]Message from BossmanJack[/]"); _sendChatMessage($"[img]{_config.KickIcon}[/img] BossmanJack: {e.Content.TranslateKickEmotes()}"); } } private void OnUsersJoined(object sender, List users, UsersJsonModel jsonPayload) { _logger.Debug($"Received {users.Count} user join events"); foreach (var user in users) { if (user.Id == _config.GambaSeshUserId && _config.EnableGambaSeshDetect) { _gambaSeshPresent = true; } AnsiConsole.MarkupLine($"[green]{user.Username.EscapeMarkup()} joined![/]"); } } private void OnUsersParted(object sender, List userIds) { if (userIds.Contains(_config.GambaSeshUserId) && _config.EnableGambaSeshDetect) { _gambaSeshPresent = false; } _logger.Debug($"Received {userIds.Count} user part events"); foreach (var id in userIds) { AnsiConsole.MarkupLine($"[red]{id} left the chat...[/]"); } } private void OnKfWsDisconnected(object sender, DisconnectionInfo disconnectionInfo) { _logger.Error($"Sneedchat disconnected due to {disconnectionInfo.Type}"); } private void OnKfWsReconnected(object sender, ReconnectionInfo reconnectionInfo) { _logger.Error($"Sneedchat reconnected due to {reconnectionInfo.Type}"); _logger.Info($"Rejoining {_config.KfChatRoomId}"); _kfClient.JoinRoom(_config.KfChatRoomId); } private void OnPusherWsReconnected(object sender, ReconnectionInfo reconnectionInfo) { _logger.Error($"Pusher reconnected due to {reconnectionInfo.Type}"); foreach (var channel in _config.PusherChannels) { _logger.Info($"Rejoining {channel}"); _kickClient.SendPusherSubscribe(channel); } } private void OnPusherSubscriptionSucceeded(object sender, PusherModels.BasePusherEventModel? e) { _logger.Info($"Pusher indicates subscription to {e?.Channel} was successful"); } }