2024-08-12 16:42:53 +00:00
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Web;
|
|
|
|
using Microsoft.Extensions.Hosting;
|
|
|
|
using NAudio.Wave;
|
|
|
|
using NAudio.Wave.SampleProviders;
|
|
|
|
using Serilog;
|
2024-08-12 17:49:36 +00:00
|
|
|
using TwitchChatTTS.Chat.Observers;
|
2024-08-12 16:42:53 +00:00
|
|
|
using TwitchChatTTS.Chat.Speech;
|
|
|
|
|
|
|
|
namespace TwitchChatTTS
|
|
|
|
{
|
|
|
|
public class TTSListening : IHostedService
|
|
|
|
{
|
|
|
|
private readonly AudioPlaybackEngine _playback;
|
|
|
|
private readonly TTSPlayer _player;
|
2024-08-12 17:49:36 +00:00
|
|
|
private readonly TTSConsumer _consumer;
|
|
|
|
private readonly IDisposable _subscription;
|
2024-08-12 16:42:53 +00:00
|
|
|
private readonly ILogger _logger;
|
|
|
|
|
|
|
|
|
2024-08-12 17:49:36 +00:00
|
|
|
public TTSListening(AudioPlaybackEngine playback, TTSPlayer player, TTSPublisher publisher, TTSConsumer consumer, ILogger logger)
|
2024-08-12 16:42:53 +00:00
|
|
|
{
|
|
|
|
_playback = playback;
|
|
|
|
_player = player;
|
2024-08-12 17:49:36 +00:00
|
|
|
_consumer = consumer;
|
|
|
|
_subscription = publisher.Subscribe(consumer);
|
2024-08-12 16:42:53 +00:00
|
|
|
_logger = logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task StartAsync(CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
Task.Run(async () =>
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if (cancellationToken.IsCancellationRequested)
|
|
|
|
{
|
|
|
|
_logger.Warning("TTS Listening - Cancellation requested.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var group = _player.ReceiveBuffer();
|
|
|
|
if (group == null)
|
|
|
|
{
|
|
|
|
await Task.Delay(200, cancellationToken);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cancellationToken.IsCancellationRequested)
|
|
|
|
{
|
|
|
|
_logger.Warning("TTS Listening - Cancellation requested.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-12-02 21:00:50 +00:00
|
|
|
if (DateTime.UtcNow - group.Timestamp > TimeSpan.FromSeconds(60))
|
|
|
|
{
|
|
|
|
_logger.Debug("Ignored message due to being in queue for too long.");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-08-12 16:42:53 +00:00
|
|
|
FetchMasterAudio(group);
|
|
|
|
}
|
|
|
|
catch (COMException e)
|
|
|
|
{
|
|
|
|
_logger.Error(e, "Failed to send request for TTS [HResult: " + e.HResult + "]");
|
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
|
|
|
_logger.Error(e, "Failed to send request for TTS.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return Task.CompletedTask;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Task StopAsync(CancellationToken cancellationToken)
|
|
|
|
{
|
2024-08-12 17:49:36 +00:00
|
|
|
_subscription?.Dispose();
|
2024-08-12 16:42:53 +00:00
|
|
|
return Task.CompletedTask;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Task FetchMasterAudio(TTSGroupedMessage group)
|
|
|
|
{
|
|
|
|
return Task.Run(() =>
|
|
|
|
{
|
|
|
|
var list = new List<ISampleProvider>();
|
|
|
|
foreach (var message in group.Messages)
|
|
|
|
{
|
|
|
|
if (string.IsNullOrEmpty(message.Message))
|
|
|
|
{
|
|
|
|
using (var reader = new AudioFileReader(message.File))
|
|
|
|
{
|
|
|
|
var data = _playback.ConvertSound(reader.ToWaveProvider());
|
|
|
|
var resampled = new WdlResamplingSampleProvider(data, _playback.SampleRate);
|
|
|
|
list.Add(resampled);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (string.IsNullOrEmpty(message.Voice))
|
|
|
|
{
|
|
|
|
_logger.Error($"No voice has been selected for this message [message: {message.Message}]");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
string url = $"https://api.streamelements.com/kappa/v2/speech?voice={message.Voice}&text={HttpUtility.UrlEncode(message.Message.Trim())}";
|
|
|
|
var nws = new NetworkWavSound(url);
|
|
|
|
var provider = new CachedWavProvider(nws);
|
|
|
|
var data = _playback.ConvertSound(provider);
|
|
|
|
var resampled = new WdlResamplingSampleProvider(data, _playback.SampleRate);
|
|
|
|
list.Add(resampled);
|
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
2024-08-14 18:19:18 +00:00
|
|
|
_logger.Error(e, $"Failed to fetch TTS message [message: {message.Message.Trim()}][chatter id: {group.ChatterId}].");
|
2024-08-12 16:42:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var merged = new ConcatenatingSampleProvider(list);
|
|
|
|
group.Audio = merged;
|
|
|
|
_player.Ready(group);
|
2024-08-12 17:49:36 +00:00
|
|
|
_consumer.OnNext(group);
|
2024-08-12 16:42:53 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|