unison/Handlers/MPDHandler.cs

612 lines
20 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Net;
using System.Text.RegularExpressions;
using System.Threading;
2021-08-16 00:13:47 +00:00
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Media.Imaging;
2022-11-02 23:39:03 +00:00
using System.Windows.Threading;
using MpcNET;
using MpcNET.Commands.Database;
using MpcNET.Commands.Playback;
2021-10-01 00:24:18 +00:00
using MpcNET.Commands.Queue;
2021-10-03 15:20:14 +00:00
using MpcNET.Commands.Reflection;
using MpcNET.Commands.Status;
using MpcNET.Message;
using MpcNET.Types;
2021-08-16 00:13:47 +00:00
namespace unison
{
2021-10-05 12:18:58 +00:00
public class Statistics
{
public int Songs { get; set; }
public int Albums { get; set; }
public int Artists { get; set; }
public string TotalPlaytime { get; set; }
public string Uptime { get; set; }
public string TotalTimePlayed { get; set; }
public string DatabaseUpdate { get; set; }
}
public class MPDHandler
{
2021-09-01 00:35:37 +00:00
private bool _connected;
public string _version;
2021-09-01 00:35:37 +00:00
private int _currentVolume;
2021-10-04 17:20:41 +00:00
private int _previousVolume;
2021-09-01 00:35:37 +00:00
private bool _currentRandom;
private bool _currentRepeat;
private bool _currentSingle;
private bool _currentConsume;
private double _currentTime;
private double _totalTime;
private MpdStatus _currentStatus;
private IMpdFile _currentSong;
2022-11-05 00:05:46 +00:00
private BitmapImage _cover;
2021-10-05 12:18:58 +00:00
public Statistics _stats;
private readonly System.Timers.Timer _elapsedTimer;
2022-11-02 23:39:03 +00:00
private DispatcherTimer _retryTimer;
2021-09-01 00:35:37 +00:00
2021-09-02 22:21:15 +00:00
bool _isUpdatingStatus = false;
bool _isUpdatingSong = false;
2022-04-07 23:39:59 +00:00
public IPAddress _ipAddress;
2021-09-01 00:35:37 +00:00
private event EventHandler ConnectionChanged;
private event EventHandler StatusChanged;
private event EventHandler SongChanged;
private event EventHandler CoverChanged;
private MpcConnection _connection;
private MpcConnection _commandConnection;
private IPEndPoint _mpdEndpoint;
2022-11-02 23:19:35 +00:00
private CancellationTokenSource _cancelCommand;
private CancellationTokenSource _cancelConnect;
2021-08-16 00:13:47 +00:00
public MPDHandler()
{
2022-11-02 23:39:03 +00:00
Startup(null, null);
2021-10-05 12:18:58 +00:00
_stats = new Statistics();
2022-11-02 23:39:03 +00:00
_retryTimer = new DispatcherTimer();
_retryTimer.Interval = TimeSpan.FromSeconds(5);
_retryTimer.Tick += Startup;
_elapsedTimer = new System.Timers.Timer(500);
_elapsedTimer.Elapsed += new System.Timers.ElapsedEventHandler(ElapsedTimer);
ConnectionChanged += OnConnectionChanged;
StatusChanged += OnStatusChanged;
SongChanged += OnSongChanged;
CoverChanged += OnCoverChanged;
}
2021-09-01 00:35:37 +00:00
private void ElapsedTimer(object sender, System.Timers.ElapsedEventArgs e)
{
if ((_currentTime < _totalTime || _totalTime == -1) && (_currentStatus.State == MpdState.Play))
_currentTime += 0.5;
else
_elapsedTimer.Stop();
}
void OnConnectionChanged(object sender, EventArgs e)
{
2022-11-02 23:39:03 +00:00
if (!_connected)
_retryTimer.Start();
else
_retryTimer.Stop();
Application.Current.Dispatcher.Invoke(() =>
{
MainWindow MainWin = (MainWindow)Application.Current.MainWindow;
MainWin.OnConnectionChanged(sender, e);
SnapcastHandler Snapcast = (SnapcastHandler)Application.Current.Properties["snapcast"];
Snapcast.OnConnectionChanged(sender, e);
2021-09-01 00:35:37 +00:00
});
}
static void OnStatusChanged(object sender, EventArgs e)
{
Application.Current.Dispatcher.Invoke(() =>
{
MainWindow MainWin = (MainWindow)Application.Current.MainWindow;
MainWin.OnStatusChanged(sender, e);
2021-09-01 00:35:37 +00:00
});
}
static void OnSongChanged(object sender, EventArgs e)
{
Application.Current.Dispatcher.Invoke(() =>
{
MainWindow MainWin = (MainWindow)Application.Current.MainWindow;
MainWin.OnSongChanged(sender, e);
2021-09-01 00:35:37 +00:00
});
}
static void OnCoverChanged(object sender, EventArgs e)
{
Application.Current.Dispatcher.Invoke(() =>
{
MainWindow MainWin = (MainWindow)Application.Current.MainWindow;
MainWin.OnCoverChanged(sender, e);
2021-09-01 00:35:37 +00:00
});
}
2021-09-02 22:21:15 +00:00
public void SendCommand<T>(IMpcCommand<T> command)
{
Task.Run(async () =>
{
await SafelySendCommandAsync(command);
});
}
public async Task<T> SafelySendCommandAsync<T>(IMpcCommand<T> command)
{
2022-11-02 23:19:35 +00:00
if (_commandConnection == null || !IsConnected())
2021-09-02 22:21:15 +00:00
{
Trace.WriteLine("[SafelySendCommandAsync] no command connection");
return default(T);
}
try
{
IMpdMessage<T> response = await _commandConnection.SendAsync(command);
if (!response.IsResponseValid)
{
string mpdError = response.Response?.Result?.MpdError;
2021-09-02 22:21:15 +00:00
if (mpdError != null && mpdError != "")
throw new Exception(mpdError);
else
throw new Exception($"Invalid server response: {response}.");
}
return response.Response.Content;
}
catch (Exception e)
{
Trace.WriteLine($"Sending {command.GetType().Name} failed: {e.Message}");
}
return default(T);
}
2022-11-02 23:39:03 +00:00
public async void Startup(object sender, EventArgs e)
{
2022-11-02 23:19:35 +00:00
await Initialize();
}
public async Task Initialize()
{
Trace.WriteLine("Initializing");
Disconnected();
if (!_connected)
2022-11-02 23:19:35 +00:00
await Connect();
}
public void Disconnected()
{
_connected = false;
ConnectionChanged?.Invoke(this, EventArgs.Empty);
_commandConnection?.DisconnectAsync();
_connection?.DisconnectAsync();
_cancelConnect?.Cancel();
_cancelConnect = new CancellationTokenSource();
_cancelCommand?.Cancel();
_cancelCommand = new CancellationTokenSource();
_connection = null;
_commandConnection = null;
Trace.WriteLine("Disconnected");
}
2022-11-02 23:19:35 +00:00
public async Task Connect()
{
2022-11-02 23:19:35 +00:00
Trace.WriteLine("Connecting");
if (_cancelCommand.IsCancellationRequested || _cancelConnect.IsCancellationRequested)
return;
try
{
2022-11-02 23:19:35 +00:00
_connection = await ConnectInternal(_cancelConnect.Token);
_commandConnection = await ConnectInternal(_cancelCommand.Token);
}
2022-04-07 23:39:59 +00:00
catch (MpcNET.Exceptions.MpcConnectException e)
{
_connected = false;
2022-04-07 23:39:59 +00:00
Trace.WriteLine($"Error in connect: {e.Message}");
ConnectionChanged?.Invoke(this, EventArgs.Empty);
2022-04-07 23:39:59 +00:00
return;
}
if (_connection != null && _commandConnection != null)
{
if (_connection.IsConnected && _commandConnection.IsConnected)
{
_connected = true;
_version = _connection.Version;
ConnectionChanged?.Invoke(this, EventArgs.Empty);
}
}
else
{
_connected = false;
ConnectionChanged?.Invoke(this, EventArgs.Empty);
return;
}
2021-08-16 00:13:47 +00:00
await UpdateStatusAsync();
await UpdateSongAsync();
2022-11-02 23:19:35 +00:00
Loop(_cancelCommand.Token);
}
private async Task<MpcConnection> ConnectInternal(CancellationToken token)
{
2022-11-02 23:19:35 +00:00
if (token.IsCancellationRequested)
return null;
2022-04-07 23:39:59 +00:00
IPAddress.TryParse(Properties.Settings.Default.mpd_host, out _ipAddress);
2022-04-07 23:39:59 +00:00
if (_ipAddress == null)
{
try
{
2022-04-07 23:39:59 +00:00
IPAddress[] addrList = Dns.GetHostAddresses(Properties.Settings.Default.mpd_host);
if (addrList.Length > 0)
2022-04-07 23:39:59 +00:00
{
foreach (IPAddress addr in addrList)
{
if (addr.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork)
_ipAddress = addr;
}
}
}
catch (Exception)
{
throw new MpcNET.Exceptions.MpcConnectException("No correct IP provided by user.");
}
}
2022-04-07 23:39:59 +00:00
_mpdEndpoint = new IPEndPoint(_ipAddress, Properties.Settings.Default.mpd_port);
MpcConnection connection = new MpcConnection(_mpdEndpoint);
await connection.ConnectAsync(token);
2022-04-12 22:43:04 +00:00
if (!string.IsNullOrEmpty(Properties.Settings.Default.mpd_password))
{
IMpdMessage<string> result = await connection.SendAsync(new PasswordCommand(Properties.Settings.Default.mpd_password));
if (!result.IsResponseValid)
{
string mpdError = result.Response?.Result?.MpdError;
2021-09-01 00:35:37 +00:00
Trace.WriteLine(mpdError);
}
2022-04-12 22:43:04 +00:00
}
return connection;
}
private void Loop(CancellationToken token)
{
Task.Run(async () =>
{
while (true)
{
try
{
2022-11-02 23:19:35 +00:00
if (token.IsCancellationRequested || _connection == null || !IsConnected())
break;
IMpdMessage<string> idleChanges = await _connection.SendAsync(new IdleCommand("stored_playlist playlist player mixer output options update"));
if (idleChanges.IsResponseValid)
await HandleIdleResponseAsync(idleChanges.Response.Content);
else
2022-04-07 23:39:59 +00:00
{
2022-11-02 23:19:35 +00:00
Trace.WriteLine($"Error in Idle connection thread (1): {idleChanges.Response?.Content}");
throw new Exception(idleChanges.Response?.Content);
2022-04-07 23:39:59 +00:00
}
}
catch (Exception e)
{
2022-11-02 23:19:35 +00:00
if (token.IsCancellationRequested)
Trace.WriteLine($"Idle connection cancelled.");
else
Trace.WriteLine($"Error in Idle connection thread: {e.Message}");
await Initialize();
break;
}
}
}, token).ConfigureAwait(false);
}
private async Task HandleIdleResponseAsync(string subsystems)
{
try
{
if (subsystems.Contains("player") || subsystems.Contains("mixer") || subsystems.Contains("output") || subsystems.Contains("options"))
{
await UpdateStatusAsync();
if (subsystems.Contains("player"))
await UpdateSongAsync();
}
}
catch (Exception e)
{
Trace.WriteLine($"Error in Idle connection thread: {e.Message}");
2022-11-02 23:19:35 +00:00
await Initialize();
}
}
private async Task UpdateStatusAsync()
{
2021-09-02 22:21:15 +00:00
if (_connection == null || _isUpdatingStatus)
return;
_isUpdatingStatus = true;
try
{
IMpdMessage<MpdStatus> response = await _connection.SendAsync(new StatusCommand());
if (response != null && response.IsResponseValid)
{
2021-09-01 00:35:37 +00:00
_currentStatus = response.Response.Content;
UpdateStatus();
}
else
throw new Exception();
}
catch (Exception e)
{
Trace.WriteLine($"Error in Idle connection thread: {e.Message}");
2022-11-02 23:19:35 +00:00
await Initialize();
2021-08-16 00:13:47 +00:00
}
2021-09-02 22:21:15 +00:00
_isUpdatingStatus = false;
2021-08-16 00:13:47 +00:00
}
private async Task UpdateSongAsync()
2021-08-16 00:13:47 +00:00
{
2022-11-02 23:19:35 +00:00
Trace.WriteLine("Updating song");
2021-09-02 22:21:15 +00:00
if (_connection == null || _isUpdatingSong)
return;
_isUpdatingSong = true;
try
{
IMpdMessage<IMpdFile> response = await _connection.SendAsync(new CurrentSongCommand());
if (response != null && response.IsResponseValid)
{
2021-09-01 00:35:37 +00:00
_currentSong = response.Response.Content;
UpdateSong();
}
else
throw new Exception();
}
catch (Exception e)
{
Trace.WriteLine($"Error in Idle connection thread: {e.Message}");
2022-11-02 23:19:35 +00:00
await Initialize();
}
2021-09-02 22:21:15 +00:00
_isUpdatingSong = false;
}
2022-11-02 23:19:35 +00:00
private async void GetAlbumCover(string path, CancellationToken token)
{
2022-11-02 23:19:35 +00:00
if (token.IsCancellationRequested)
return;
List<byte> data = new List<byte>();
try
{
bool ReadPictureFailed = true;
long totalBinarySize = 9999;
long currentSize = 0;
do
{
2022-03-30 22:27:33 +00:00
if (_connection == null)
return;
IMpdMessage<MpdBinaryData> albumReq = await _connection.SendAsync(new AlbumArtCommand(path, currentSize));
if (!albumReq.IsResponseValid)
break;
MpdBinaryData response = albumReq.Response.Content;
if (response == null || response.Binary == 0)
break;
ReadPictureFailed = false;
totalBinarySize = response.Size;
currentSize += response.Binary;
data.AddRange(response.Data);
}
while (currentSize < totalBinarySize && !token.IsCancellationRequested);
do
{
if (!ReadPictureFailed)
break;
if (_connection == null)
return;
IMpdMessage<MpdBinaryData> albumReq = await _connection.SendAsync(new ReadPictureCommand(path, currentSize));
if (!albumReq.IsResponseValid)
break;
MpdBinaryData response = albumReq.Response.Content;
if (response == null || response.Binary == 0)
break;
totalBinarySize = response.Size;
currentSize += response.Binary;
data.AddRange(response.Data);
}
while (currentSize < totalBinarySize && !token.IsCancellationRequested);
}
catch (Exception e)
{
2022-11-02 23:19:35 +00:00
if (token.IsCancellationRequested)
return;
2021-09-01 00:35:37 +00:00
Trace.WriteLine("Exception caught while getting albumart: " + e);
return;
}
if (data.Count == 0)
_cover = null;
else
{
using MemoryStream stream = new MemoryStream(data.ToArray());
2022-11-05 00:05:46 +00:00
_cover = new BitmapImage();
_cover.BeginInit();
_cover.CacheOption = BitmapCacheOption.OnLoad;
_cover.StreamSource = stream;
_cover.EndInit();
_cover.Freeze();
}
UpdateCover();
2021-08-16 00:13:47 +00:00
}
2021-09-02 22:21:15 +00:00
public void UpdateStatus()
2021-08-16 00:13:47 +00:00
{
2021-09-02 22:21:15 +00:00
if (!_connected || _currentStatus == null)
2021-08-16 00:13:47 +00:00
return;
2021-09-02 22:21:15 +00:00
_currentRandom = _currentStatus.Random;
_currentRepeat = _currentStatus.Repeat;
_currentConsume = _currentStatus.Consume;
_currentSingle = _currentStatus.Single;
_currentVolume = _currentStatus.Volume;
StatusChanged?.Invoke(this, EventArgs.Empty);
}
public void UpdateSong()
{
if (!_connected || _currentSong == null)
return;
2021-08-16 00:13:47 +00:00
2021-09-01 00:35:37 +00:00
_currentTime = _currentStatus.Elapsed.TotalSeconds;
_totalTime = _currentSong.Time;
if (!_elapsedTimer.Enabled)
_elapsedTimer.Start();
SongChanged?.Invoke(this, EventArgs.Empty);
2021-09-01 00:35:37 +00:00
string uri = Regex.Escape(_currentSong.Path);
2022-11-02 23:19:35 +00:00
GetAlbumCover(uri, _cancelCommand.Token);
}
public void UpdateCover()
{
CoverChanged?.Invoke(this, EventArgs.Empty);
}
2021-09-01 00:35:37 +00:00
public IMpdFile GetCurrentSong() => _currentSong;
public MpdStatus GetStatus() => _currentStatus;
2022-11-05 00:05:46 +00:00
public BitmapImage GetCover() => _cover;
public string GetVersion() => _version;
2021-10-05 12:18:58 +00:00
public Statistics GetStats() => _stats;
2021-09-01 00:35:37 +00:00
public double GetCurrentTime() => _currentTime;
public bool IsConnected() => _connected;
public bool IsPlaying() => _currentStatus?.State == MpdState.Play;
2021-08-16 00:13:47 +00:00
public void Prev() => SendCommand(new PreviousCommand());
public void Next() => SendCommand(new NextCommand());
public void PlayPause() => SendCommand(new PauseResumeCommand());
2021-08-16 00:13:47 +00:00
public void Random() => SendCommand(new RandomCommand(!_currentRandom));
public void Repeat() => SendCommand(new RepeatCommand(!_currentRepeat));
public void Single() => SendCommand(new SingleCommand(!_currentSingle));
public void Consume() => SendCommand(new ConsumeCommand(!_currentConsume));
2021-08-16 00:13:47 +00:00
public void SetTime(double value) => SendCommand(new SeekCurCommand(value));
public void SetVolume(int value) => SendCommand(new SetVolumeCommand((byte)value));
2021-09-01 00:35:37 +00:00
public void VolumeUp()
{
_currentVolume += Properties.Settings.Default.volume_offset;
if (_currentVolume > 100)
_currentVolume = 100;
SetVolume(_currentVolume);
}
2021-08-16 00:13:47 +00:00
2021-09-01 00:35:37 +00:00
public void VolumeDown()
{
_currentVolume -= Properties.Settings.Default.volume_offset;
if (_currentVolume < 0)
_currentVolume = 0;
SetVolume(_currentVolume);
}
2021-10-01 00:24:18 +00:00
2021-10-04 17:20:41 +00:00
public void VolumeMute()
{
if (_currentVolume == 0)
{
_currentVolume = _previousVolume;
_previousVolume = 0;
}
else
{
_previousVolume = _currentVolume;
_currentVolume = 0;
}
SetVolume(_currentVolume);
}
2021-10-01 00:24:18 +00:00
public void ClearQueue() => SendCommand(new ClearCommand());
public void PlayCommand() => SendCommand(new PlayCommand(0));
public void AddSong(string Uri)
{
SendCommand(new AddCommand(Uri));
}
2021-10-03 15:20:14 +00:00
public void ClearAddAndPlay(string Uri)
{
CommandList commandList = new CommandList(new IMpcCommand<object>[] { new ClearCommand(), new AddCommand(Uri), new PlayCommand(0) });
SendCommand(commandList);
}
2021-10-05 12:18:58 +00:00
public async void QueryStats()
{
Dictionary<string, string> response = await SafelySendCommandAsync(new StatsCommand());
if (response != null)
{
_stats.Songs = int.Parse(response["songs"]);
_stats.Albums = int.Parse(response["albums"]);
_stats.Artists = int.Parse(response["artists"]);
TimeSpan time;
time = TimeSpan.FromSeconds(int.Parse(response["uptime"]));
_stats.Uptime = time.ToString(@"dd\:hh\:mm\:ss");
time = TimeSpan.FromSeconds(int.Parse(response["db_playtime"]));
_stats.TotalPlaytime = time.ToString(@"dd\:hh\:mm\:ss");
time = TimeSpan.FromSeconds(int.Parse(response["playtime"]));
_stats.TotalTimePlayed = time.ToString(@"dd\:hh\:mm\:ss");
DateTime date = new DateTime(1970, 1, 1).AddSeconds(int.Parse(response["db_update"])).ToLocalTime();
_stats.DatabaseUpdate = date.ToString("dd/MM/yyyy @ HH:mm");
}
2021-10-05 12:18:58 +00:00
}
}
2021-09-01 00:35:37 +00:00
}