using System.Text.Json; using CommonSocketLibrary.Abstract; using CommonSocketLibrary.Common; using Serilog; using TwitchChatTTS.OBS.Socket.Data; namespace TwitchChatTTS.OBS.Socket.Handlers { public class RequestResponseHandler : IWebSocketHandler { private readonly ILogger _logger; public int OperationCode { get; } = 7; public RequestResponseHandler( ILogger logger ) { _logger = logger; } public Task Execute(SocketClient sender, Data data) { if (data is not RequestResponseMessage message || message == null) return Task.CompletedTask; if (sender is not OBSSocketClient obs) return Task.CompletedTask; _logger.Debug($"Received an OBS request response [obs request id: {message.RequestId}]"); var requestData = obs.Take(message.RequestId); if (requestData == null) { _logger.Warning($"OBS Request Response not being processed: request not stored [obs request id: {message.RequestId}]"); return Task.CompletedTask; } var request = requestData.Message; if (request == null) return Task.CompletedTask; try { switch (request.RequestType) { case "GetOutputStatus": _logger.Debug($"Fetched stream's live status [live: {obs.Streaming}][obs request id: {message.RequestId}]"); break; case "GetSceneItemId": { if (!request.RequestData.TryGetValue("sceneName", out object? sceneName) || sceneName == null) { _logger.Warning($"Failed to find the scene name that was requested [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!request.RequestData.TryGetValue("sourceName", out object? sourceName) || sourceName == null) { _logger.Warning($"Failed to find the scene item name that was requested [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (message.ResponseData == null) { _logger.Warning($"OBS Response is null [scene: {sceneName}][scene item: {sourceName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!message.ResponseData.TryGetValue("sceneItemId", out object? sceneItemId) || sceneItemId == null) { _logger.Warning($"Failed to fetch the scene item id [scene: {sceneName}][scene item: {sourceName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Found the scene item by name [scene: {sceneName}][source: {sourceName}][id: {sceneItemId}][obs request id: {message.RequestId}]."); obs.AddSourceId(sourceName.ToString()!, (long)sceneItemId); requestData.ResponseValues = message.ResponseData; break; } case "GetSceneItemTransform": { if (!request.RequestData.TryGetValue("sceneName", out object? sceneName) || sceneName == null) { _logger.Warning($"Failed to find the scene name that was requested [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!request.RequestData.TryGetValue("sceneItemId", out object? sceneItemId) || sceneItemId == null) { _logger.Warning($"Failed to find the scene item name that was requested [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (message.ResponseData == null) { _logger.Warning($"OBS Response is null [scene: {sceneName}][scene item id: {sceneItemId}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!message.ResponseData.TryGetValue("sceneItemTransform", out object? transformData) || transformData == null) { _logger.Warning($"Failed to fetch the OBS transformation data [obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Fetched OBS transformation data [scene: {sceneName}][scene item id: {sceneItemId}][transformation: {transformData}][obs request id: {message.RequestId}]"); requestData.ResponseValues = message.ResponseData; break; } case "GetSceneItemEnabled": { if (!request.RequestData.TryGetValue("sceneName", out object? sceneName) || sceneName == null) { _logger.Warning($"Failed to find the scene name that was requested [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!request.RequestData.TryGetValue("sceneItemId", out object? sceneItemId) || sceneItemId == null) { _logger.Warning($"Failed to find the scene item name that was requested [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (message.ResponseData == null) { _logger.Warning($"OBS Response is null [scene: {sceneName}][scene item id: {sceneItemId}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!message.ResponseData.TryGetValue("sceneItemEnabled", out object? sceneItemVisibility) || sceneItemVisibility == null) { _logger.Warning($"Failed to fetch the scene item visibility [scene: {sceneName}][scene item id: {sceneItemId}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Fetched OBS scene item visibility [scene: {sceneName}][scene item id: {sceneItemId}][visibility: {sceneItemVisibility}][obs request id: {message.RequestId}]"); requestData.ResponseValues = message.ResponseData; break; } case "SetSceneItemTransform": { if (!request.RequestData.TryGetValue("sceneName", out object? sceneName) || sceneName == null) { _logger.Warning($"Failed to find the scene name that was requested [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!request.RequestData.TryGetValue("sceneItemId", out object? sceneItemId) || sceneItemId == null) { _logger.Warning($"Failed to find the scene item name that was requested [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Received response from OBS for updating scene item transformation [scene: {sceneName}][scene item id: {sceneItemId}][obs request id: {message.RequestId}]"); break; } case "SetSceneItemEnabled": { if (!request.RequestData.TryGetValue("sceneName", out object? sceneName) || sceneName == null) { _logger.Warning($"Failed to find the scene name that was requested [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!request.RequestData.TryGetValue("sceneItemId", out object? sceneItemId) || sceneItemId == null) { _logger.Warning($"Failed to find the scene item name that was requested [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Received response from OBS for updating scene item visibility [scene: {sceneName}][scene item id: {sceneItemId}][obs request id: {message.RequestId}]"); break; } case "GetGroupList": { if (message.ResponseData == null) { _logger.Warning($"OBS Response is null [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!message.ResponseData.TryGetValue("groups", out object? value) || value == null) { _logger.Warning($"Failed to fetch the scene item visibility [obs request id: {message.RequestId}]"); return Task.CompletedTask; } var groups = JsonSerializer.Deserialize>(value.ToString()); _logger.Debug($"Fetched OBS groups [obs request id: {message.RequestId}]"); requestData.ResponseValues = new Dictionary() { { "groups", groups } }; break; } case "GetGroupSceneItemList": { if (!request.RequestData.TryGetValue("sceneName", out object? sceneName) || sceneName == null) { _logger.Warning($"Failed to find the scene name that was requested [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (message.ResponseData == null) { _logger.Warning($"OBS Response is null [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!message.ResponseData.TryGetValue("sceneItems", out object? value) || value == null) { _logger.Warning($"Failed to fetch the scene item visibility [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Fetched OBS scene items in group [scene: {sceneName}][obs request id: {message.RequestId}]"); var sceneItems = JsonSerializer.Deserialize>(value.ToString()!, new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase }); if (sceneItems == null) { _logger.Warning($"Failed to deserialize the data received [scene: {sceneName}][obs request id: {message.RequestId}]"); return Task.CompletedTask; } foreach (var sceneItem in sceneItems) obs.AddSourceId(sceneItem.SourceName, sceneItem.SceneItemId); requestData.ResponseValues = new Dictionary() { { "groups", sceneItems } }; break; } case "Sleep": { if (!request.RequestData.TryGetValue("sleepMillis", out object? sleepMillis) || sleepMillis == null) { _logger.Warning($"Failed to find the amount of time to sleep for [obs request id: {message.RequestId}]"); return Task.CompletedTask; } _logger.Debug($"Received response from OBS for sleeping [sleep: {sleepMillis}][obs request id: {message.RequestId}]"); break; } case "GetStreamStatus": { if (message.ResponseData == null) { _logger.Warning($"OBS Response is null [obs request id: {message.RequestId}]"); return Task.CompletedTask; } if (!message.ResponseData.TryGetValue("outputActive", out object? outputActive) || outputActive == null) { _logger.Warning($"Failed to fetch the scene item visibility [obs request id: {message.RequestId}]"); return Task.CompletedTask; } obs.Streaming = outputActive?.ToString()!.ToLower() == "true"; requestData.ResponseValues = message.ResponseData; _logger.Information($"OBS is currently {(obs.Streaming ? "" : "not ")}streaming."); break; } default: _logger.Warning($"OBS Request Response not being processed [type: {request.RequestType}][{string.Join(Environment.NewLine, message.ResponseData?.Select(kvp => kvp.Key + " = " + kvp.Value?.ToString()) ?? [])}]"); break; } } catch (Exception ex) { _logger.Error(ex, $"Failed to process the response from OBS for a request [type: {request.RequestType}]"); } finally { if (requestData.Callback != null) requestData.Callback(requestData.ResponseValues); } return Task.CompletedTask; } } }