mirror of
https://github.com/9ParsonsB/Pulsar.git
synced 2025-04-05 17:39:39 -04:00
* Expose core error logger to plugins and report custom criteria errors Fixes #77 This adds an error logging method on the IObservatoryCore interface that writes the exception details to ObservatoryCore's central error log (found in `${Documents}/ObservatoryErrorLog.txt`). In addition, added a timestamp to each error log. Also updates the Explorer to report Custom Criteria file load errors and execution errors to the log. Also updates HeraldNotifier to report CacheIndex.json parse failures to the error log as well. * Expand debugging/error logging in Herald; cleanup empty mp3 files Herald crashes if attempting to play 0-byte mp3s so if detected, delete, re-request (empty files can occur in some azure failure cases (ie. out of quota). Trap and log errors in other places in HeraldQueue to avoid hard crashes due to weird and wonderful unexpected stuff.
138 lines
4.6 KiB
C#
138 lines
4.6 KiB
C#
using Observatory.Framework;
|
|
using System.Collections.Generic;
|
|
using System.Threading.Tasks;
|
|
using System.Linq;
|
|
using NetCoreAudio;
|
|
using System.Threading;
|
|
using System;
|
|
using System.Diagnostics;
|
|
|
|
namespace Observatory.Herald
|
|
{
|
|
class HeraldQueue
|
|
{
|
|
private Queue<NotificationArgs> notifications;
|
|
private bool processing;
|
|
private string voice;
|
|
private string style;
|
|
private string rate;
|
|
private byte volume;
|
|
private SpeechRequestManager speechManager;
|
|
private Player audioPlayer;
|
|
private Action<Exception, String> ErrorLogger;
|
|
|
|
public HeraldQueue(SpeechRequestManager speechManager, Action<Exception, String> errorLogger)
|
|
{
|
|
this.speechManager = speechManager;
|
|
processing = false;
|
|
notifications = new();
|
|
audioPlayer = new();
|
|
ErrorLogger = errorLogger;
|
|
}
|
|
|
|
|
|
internal void Enqueue(NotificationArgs notification, string selectedVoice, string selectedStyle = "", string selectedRate = "", int volume = 75)
|
|
{
|
|
voice = selectedVoice;
|
|
style = selectedStyle;
|
|
rate = selectedRate;
|
|
// Ignore invalid values; assume default.
|
|
volume = volume >= 0 && volume <= 100 ? volume : 75;
|
|
|
|
// Volume is perceived logarithmically, convert to exponential curve
|
|
// to make perceived volume more in line with value set.
|
|
this.volume = ((byte)System.Math.Floor(System.Math.Pow(volume / 100.0, 2.0) * 100));
|
|
|
|
notifications.Enqueue(notification);
|
|
|
|
if (!processing)
|
|
{
|
|
processing = true;
|
|
ProcessQueueAsync();
|
|
}
|
|
}
|
|
|
|
private async void ProcessQueueAsync()
|
|
{
|
|
await Task.Factory.StartNew(ProcessQueue);
|
|
}
|
|
|
|
private void ProcessQueue()
|
|
{
|
|
|
|
NotificationArgs notification = null;
|
|
try
|
|
{
|
|
while (notifications.Any())
|
|
{
|
|
audioPlayer.SetVolume(volume).Wait();
|
|
notification = notifications.Dequeue();
|
|
Debug.WriteLine("Processing notification: {0} - {1}", notification.Title, notification.Detail);
|
|
|
|
Task<string>[] audioRequestTasks = new Task<string>[2];
|
|
|
|
if (string.IsNullOrWhiteSpace(notification.TitleSsml))
|
|
{
|
|
audioRequestTasks[0] = RetrieveAudioToFile(notification.Title);
|
|
}
|
|
else
|
|
{
|
|
audioRequestTasks[0] = RetrieveAudioSsmlToFile(notification.TitleSsml);
|
|
}
|
|
|
|
if (string.IsNullOrWhiteSpace(notification.DetailSsml))
|
|
{
|
|
audioRequestTasks[1] = RetrieveAudioToFile(notification.Detail);
|
|
}
|
|
else
|
|
{
|
|
audioRequestTasks[1] = RetrieveAudioSsmlToFile(notification.DetailSsml);
|
|
}
|
|
|
|
PlayAudioRequestsSequentially(audioRequestTasks);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Debug.WriteLine($"Failed to fetch/play notification: {notification?.Title} - {notification?.Detail}");
|
|
ErrorLogger(ex, "while retrieving and playing audio for a notification");
|
|
}
|
|
finally
|
|
{
|
|
processing = false;
|
|
}
|
|
}
|
|
|
|
private async Task<string> RetrieveAudioToFile(string text)
|
|
{
|
|
return await RetrieveAudioSsmlToFile($"<speak version=\"1.0\" xmlns=\"http://www.w3.org/2001/10/synthesis\" xml:lang=\"en-US\"><voice name=\"\">{text}</voice></speak>");
|
|
}
|
|
|
|
private async Task<string> RetrieveAudioSsmlToFile(string ssml)
|
|
{
|
|
return await speechManager.GetAudioFileFromSsml(ssml, voice, style, rate);
|
|
}
|
|
|
|
private void PlayAudioRequestsSequentially(Task<string>[] requestTasks)
|
|
{
|
|
foreach (var request in requestTasks)
|
|
{
|
|
string file = request.Result;
|
|
try
|
|
{
|
|
Debug.WriteLine($"Playing audio file: {file}");
|
|
audioPlayer.Play(file).Wait();
|
|
} catch (Exception ex)
|
|
{
|
|
Debug.WriteLine($"Failed to play {file}: {ex.Message}");
|
|
ErrorLogger(ex, $"while playing: {file}");
|
|
}
|
|
|
|
while (audioPlayer.Playing)
|
|
Thread.Sleep(50);
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|