mirror of
https://kevinblog.sytes.net/Code/Jibo-Revival-Group/JiboExperiments.git
synced 2026-06-16 22:36:43 +00:00
Add GLSM listener telemetry and stale-listen recovery
This commit is contained in:
@@ -101,4 +101,49 @@ public sealed class FileTurnTelemetrySinkTests
|
||||
s => s.RecordTranscriptError(It.IsAny<Exception>(), It.IsAny<string>(), It.IsAny<CancellationToken>()),
|
||||
Times.Once());
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task HandleContext_EmitsGlsmPhaseTransitionDiagnostic()
|
||||
{
|
||||
var sink = new Mock<ITurnTelemetrySink>();
|
||||
sink.Setup(s => s.RecordTurnDiagnosticAsync(It.IsAny<string>(), It.IsAny<IReadOnlyDictionary<string, object?>>(), It.IsAny<CancellationToken>()))
|
||||
.Returns(Task.CompletedTask);
|
||||
var turnService = new WebSocketTurnFinalizationService(
|
||||
Mock.Of<IConversationBroker>(),
|
||||
Mock.Of<ISttStrategySelector>(),
|
||||
sink.Object);
|
||||
|
||||
var session = new CloudSession
|
||||
{
|
||||
Token = "glsm-phase-token",
|
||||
TurnState =
|
||||
{
|
||||
TransId = "trans-glsm",
|
||||
AwaitingTurnCompletion = true,
|
||||
SawListen = true,
|
||||
ListenOpenedUtc = DateTimeOffset.UtcNow - TimeSpan.FromSeconds(1)
|
||||
}
|
||||
};
|
||||
session.Metadata["glsmPhase"] = "HJ_LISTENING";
|
||||
|
||||
await turnService.HandleContextAsync(
|
||||
session,
|
||||
new WebSocketMessageEnvelope
|
||||
{
|
||||
HostName = "neo-hub.jibo.com",
|
||||
Path = "/listen",
|
||||
Kind = "neo-hub-listen",
|
||||
Text = """{"type":"CONTEXT","transID":"trans-glsm","data":{"topic":"conversation"}}"""
|
||||
},
|
||||
CancellationToken.None);
|
||||
|
||||
sink.Verify(
|
||||
s => s.RecordTurnDiagnosticAsync(
|
||||
"glsm_phase_transition",
|
||||
It.Is<IReadOnlyDictionary<string, object?>>(details =>
|
||||
details.ContainsKey("state") &&
|
||||
string.Equals(details["state"] == null ? null : details["state"]!.ToString(), "LISTENING", StringComparison.OrdinalIgnoreCase)),
|
||||
It.IsAny<CancellationToken>()),
|
||||
Times.AtLeastOnce());
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2523,6 +2523,47 @@ public sealed class JiboWebSocketServiceTests
|
||||
Assert.Null(session.LastIntent);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task StaleListenSetup_IsRecoveredWhenNextHotphraseListenArrives()
|
||||
{
|
||||
await _service.HandleMessageAsync(new WebSocketMessageEnvelope
|
||||
{
|
||||
HostName = "neo-hub.jibo.com",
|
||||
Path = "/listen",
|
||||
Kind = "neo-hub-listen",
|
||||
Token = "hub-stale-listen-token",
|
||||
Text = """{"type":"LISTEN","transID":"trans-stale-listen","data":{"hotphrase":true,"rules":["launch","globals/global_commands_launch"]}}"""
|
||||
});
|
||||
|
||||
var session = _store.FindSessionByToken("hub-stale-listen-token");
|
||||
Assert.NotNull(session);
|
||||
session.TurnState.ListenOpenedUtc = DateTimeOffset.UtcNow - TimeSpan.FromSeconds(12);
|
||||
session.TurnState.AwaitingTurnCompletion = true;
|
||||
session.TurnState.SawListen = true;
|
||||
session.TurnState.SawContext = false;
|
||||
session.TurnState.BufferedAudioBytes = 0;
|
||||
session.TurnState.BufferedAudioChunkCount = 0;
|
||||
session.TurnState.HotphraseEmptyTurnCount = 2;
|
||||
|
||||
var replies = await _service.HandleMessageAsync(new WebSocketMessageEnvelope
|
||||
{
|
||||
HostName = "neo-hub.jibo.com",
|
||||
Path = "/listen",
|
||||
Kind = "neo-hub-listen",
|
||||
Token = "hub-stale-listen-token",
|
||||
Text = """{"type":"LISTEN","transID":"trans-stale-listen","data":{"hotphrase":true,"rules":["launch","globals/global_commands_launch"]}}"""
|
||||
});
|
||||
|
||||
Assert.Empty(replies);
|
||||
Assert.True(session.TurnState.AwaitingTurnCompletion);
|
||||
Assert.True(session.TurnState.SawListen);
|
||||
Assert.False(session.TurnState.SawContext);
|
||||
Assert.Equal(0, session.TurnState.BufferedAudioBytes);
|
||||
Assert.Equal(0, session.TurnState.BufferedAudioChunkCount);
|
||||
Assert.Equal(0, session.TurnState.HotphraseEmptyTurnCount);
|
||||
Assert.True(session.TurnState.ListenOpenedUtc > DateTimeOffset.UtcNow - TimeSpan.FromSeconds(3));
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task BinaryAudio_AfterWordOfDayRightWordListen_IsIgnoredDuringCleanupWindow()
|
||||
{
|
||||
|
||||
Reference in New Issue
Block a user