using Jibo.Cloud.Application.Abstractions; using Jibo.Cloud.Application.Services; using Jibo.Cloud.Domain.Models; using Jibo.Runtime.Abstractions; using Moq; namespace Jibo.Cloud.Tests.Turn; public sealed class FileTurnTelemetrySinkTests { [Fact] public async Task RecordsTranscriptErrorOnTurnError() { var sink = new Mock(); var sttStrategySelector = new Mock(); sttStrategySelector.Setup(s => s.SelectAsync(It.IsAny(), It.IsAny())) .ThrowsAsync(new Exception("dummy")); var turnService = new WebSocketTurnFinalizationService(Mock.Of(), sttStrategySelector.Object, sink.Object ); await turnService.HandleTurnAsync(new CloudSession { TurnState = { BufferedAudioBytes = 100 } }, new WebSocketMessageEnvelope(), "dummy", CancellationToken.None); sink.Verify( s => s.RecordTranscriptError(It.IsAny(), It.IsAny(), It.IsAny()), Times.Once()); } [Fact] public async Task AutoFinalize_DoesNotFallbackImmediately_WhenSttThrows() { var sink = new Mock(); var sttStrategySelector = new Mock(); sttStrategySelector.Setup(s => s.SelectAsync(It.IsAny(), It.IsAny())) .ThrowsAsync(new InvalidOperationException("ffmpeg failed")); var turnService = new WebSocketTurnFinalizationService(Mock.Of(), sttStrategySelector.Object, sink.Object ); var session = new CloudSession { TurnState = { AwaitingTurnCompletion = true, SawListen = true, SawContext = true, BufferedAudioBytes = 12000, BufferedAudioChunkCount = 5, FirstAudioReceivedUtc = DateTimeOffset.UtcNow - TimeSpan.FromSeconds(2) } }; var replies = await turnService.HandleContextAsync( session, new WebSocketMessageEnvelope { Text = """{"type":"CONTEXT","data":{"topic":"conversation"}}""" }, CancellationToken.None); Assert.Empty(replies); Assert.True(session.TurnState.AwaitingTurnCompletion); Assert.Equal(12000, session.TurnState.BufferedAudioBytes); Assert.Equal("ffmpeg failed", session.TurnState.LastSttError); sink.Verify( s => s.RecordTranscriptError(It.IsAny(), It.IsAny(), It.IsAny()), Times.Once()); } }