using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Linq; using System.Net; using System.Text.RegularExpressions; using System.Threading; using System.Threading.Tasks; using System.Windows; using System.Windows.Media.Imaging; using System.Windows.Threading; using MpcNET; using MpcNET.Commands.Database; using MpcNET.Commands.Playback; using MpcNET.Commands.Queue; using MpcNET.Commands.Reflection; using MpcNET.Commands.Status; using MpcNET.Message; using MpcNET.Types; namespace unison { 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 { private bool _connected; private string _version; private int _currentVolume; private int _previousVolume; private bool _currentRandom; private bool _currentRepeat; private bool _currentSingle; private bool _currentConsume; private double _currentTime; private double _totalTime; private IEnumerable _Playlist; private MpdStatus _currentStatus; private IMpdFile _currentSong; private BitmapImage _cover; private readonly Statistics _stats; private readonly System.Timers.Timer _elapsedTimer; private readonly DispatcherTimer _retryTimer; private bool _isUpdatingStatus = false; private bool _isUpdatingSong = false; public IPAddress _ipAddress; 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; public CancellationTokenSource _cancelCommand; private CancellationTokenSource _cancelConnect; private bool UpdateStarted = false; public MPDHandler() { Startup(null, null); _stats = new Statistics(); _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; } 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) { 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); }); } static void OnStatusChanged(object sender, EventArgs e) { Application.Current.Dispatcher.Invoke(() => { MainWindow MainWin = (MainWindow)Application.Current.MainWindow; MainWin.OnStatusChanged(sender, e); }); } static void OnSongChanged(object sender, EventArgs e) { Application.Current.Dispatcher.Invoke(() => { MainWindow MainWin = (MainWindow)Application.Current.MainWindow; MainWin.OnSongChanged(sender, e); }); } static void OnCoverChanged(object sender, EventArgs e) { Application.Current.Dispatcher.Invoke(() => { MainWindow MainWin = (MainWindow)Application.Current.MainWindow; MainWin.OnCoverChanged(sender, e); }); } public void SendCommand(IMpcCommand command) { Task.Run(async () => { await SafelySendCommandAsync(command); }); } public async Task SafelySendCommandAsync(IMpcCommand command) { if (_commandConnection == null || !IsConnected()) { Trace.WriteLine("[SafelySendCommandAsync] no command connection"); return default; } try { IMpdMessage response = await _commandConnection.SendAsync(command); if (!response.IsResponseValid) { string mpdError = response.Response?.Result?.MpdError; 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; } public async void Startup(object sender, EventArgs e) { await Initialize(); } public async Task Initialize() { Trace.WriteLine("Initializing"); Disconnected(); if (!_connected) 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"); } public async Task Connect() { Trace.WriteLine("Connecting"); if (_cancelCommand.IsCancellationRequested || _cancelConnect.IsCancellationRequested) return; try { _connection = await ConnectInternal(_cancelConnect.Token); _commandConnection = await ConnectInternal(_cancelCommand.Token); } catch (MpcNET.Exceptions.MpcConnectException e) { _connected = false; Trace.WriteLine($"Error in connect: {e.Message}"); ConnectionChanged?.Invoke(this, EventArgs.Empty); 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; } await UpdateStatusAsync(); await UpdateSongAsync(); Loop(_cancelCommand.Token); } private async Task ConnectInternal(CancellationToken token) { if (token.IsCancellationRequested) return null; IPAddress.TryParse(Properties.Settings.Default.mpd_host, out _ipAddress); if (_ipAddress == null) { try { IPAddress[] addrList = Dns.GetHostAddresses(Properties.Settings.Default.mpd_host); if (addrList.Length > 0) { 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."); } } _mpdEndpoint = new IPEndPoint(_ipAddress, Properties.Settings.Default.mpd_port); MpcConnection connection = new MpcConnection(_mpdEndpoint); await connection.ConnectAsync(token); if (!string.IsNullOrEmpty(Properties.Settings.Default.mpd_password)) { IMpdMessage result = await connection.SendAsync(new PasswordCommand(Properties.Settings.Default.mpd_password)); if (!result.IsResponseValid) { string mpdError = result.Response?.Result?.MpdError; Trace.WriteLine(mpdError); } } return connection; } private void Loop(CancellationToken token) { Task.Run(async () => { while (true) { try { if (token.IsCancellationRequested || _connection == null || !IsConnected()) break; IMpdMessage idleChanges = await _connection.SendAsync(new IdleCommand("stored_playlist playlist player mixer output options update")); if (idleChanges.IsResponseValid) await HandleIdleResponseAsync(idleChanges.Response.Content); else { Trace.WriteLine($"Error in Idle connection thread (1): {idleChanges.Response?.Content}"); throw new Exception(idleChanges.Response?.Content); } } catch (Exception e) { 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") || subsystems.Contains("update")) { await UpdateStatusAsync(); if (subsystems.Contains("player")) await UpdateSongAsync(); if (subsystems.Contains("update")) UpdateDatabaseSync(); } } catch (Exception e) { Trace.WriteLine($"Error in Idle connection thread: {e.Message}"); await Initialize(); } } private void UpdateDatabaseSync() { if (!UpdateStarted) { Application.Current.Dispatcher.Invoke(() => { MainWindow MainWin = (MainWindow)Application.Current.MainWindow; MainWin.GetSettings().MPDDatabaseUpdate_Start(); }); UpdateStarted = true; } else { Application.Current.Dispatcher.Invoke(() => { MainWindow MainWin = (MainWindow)Application.Current.MainWindow; MainWin.GetSettings().MPDDatabaseUpdate_Stop(); MainWin.UpdateStats(); }); UpdateStarted = false; } } private async Task UpdateStatusAsync() { if (_connection == null || _isUpdatingStatus) return; _isUpdatingStatus = true; try { IMpdMessage response = await _connection.SendAsync(new StatusCommand()); if (response != null && response.IsResponseValid) { _currentStatus = response.Response.Content; UpdateStatus(); } else throw new Exception(); } catch (Exception e) { Trace.WriteLine($"Error in Idle connection thread: {e.Message}"); await Initialize(); } _isUpdatingStatus = false; } private async Task UpdateSongAsync() { Trace.WriteLine("Updating song"); if (_connection == null || _isUpdatingSong) return; _isUpdatingSong = true; try { IMpdMessage response = await _connection.SendAsync(new CurrentSongCommand()); if (response != null && response.IsResponseValid) { _currentSong = response.Response.Content; UpdateSong(); } else throw new Exception(); } catch (Exception e) { Trace.WriteLine($"Error in Idle connection thread: {e.Message}"); await Initialize(); } _isUpdatingSong = false; Trace.WriteLine("Updated song"); } private async void GetAlbumCover(string path, CancellationToken token) { if (token.IsCancellationRequested) return; List data = new List(); try { bool ReadPictureFailed = true; long totalBinarySize = 9999; long currentSize = 0; do { if (_connection == null) return; IMpdMessage 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 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) { if (token.IsCancellationRequested) return; Trace.WriteLine("Exception caught while getting albumart: " + e); return; } if (data.Count == 0) _cover = null; else { using MemoryStream stream = new MemoryStream(data.ToArray()); _cover = new BitmapImage(); _cover.BeginInit(); _cover.CacheOption = BitmapCacheOption.OnLoad; _cover.StreamSource = stream; _cover.EndInit(); _cover.Freeze(); } UpdateCover(); } public void UpdateStatus() { if (!_connected || _currentStatus == null) return; _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; _currentTime = _currentStatus.Elapsed.TotalSeconds; _totalTime = _currentSong.Time; if (!_elapsedTimer.Enabled) _elapsedTimer.Start(); SongChanged?.Invoke(this, EventArgs.Empty); string uri = Regex.Escape(_currentSong.Path); GetAlbumCover(uri, _cancelCommand.Token); } public void UpdateCover() { CoverChanged?.Invoke(this, EventArgs.Empty); } public IMpdFile GetCurrentSong() => _currentSong; public MpdStatus GetStatus() => _currentStatus; public BitmapImage GetCover() => _cover; public string GetVersion() => _version; public Statistics GetStats() => _stats; public double GetCurrentTime() => _currentTime; public IEnumerable GetPlaylist() => _Playlist; public bool IsConnected() => _connected; public bool IsPlaying() => _currentStatus?.State == MpdState.Play; public bool CanPrevNext = true; public void Prev() { if (CanPrevNext && !_isUpdatingSong) SendCommand(new PreviousCommand()); } public void Next() { if (CanPrevNext && !_isUpdatingSong) SendCommand(new NextCommand()); } public void PlayPause() => SendCommand(new PauseResumeCommand()); public void Play(int pos) => SendCommand(new PlayCommand(pos)); 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)); public void SetTime(double value) => SendCommand(new SeekCurCommand(value)); public void SetVolume(int value) => SendCommand(new SetVolumeCommand((byte)value)); public void VolumeUp() { _currentVolume += Properties.Settings.Default.volume_offset; if (_currentVolume > 100) _currentVolume = 100; SetVolume(_currentVolume); } public void VolumeDown() { _currentVolume -= Properties.Settings.Default.volume_offset; if (_currentVolume < 0) _currentVolume = 0; SetVolume(_currentVolume); } public void VolumeMute() { if (_currentVolume == 0) { _currentVolume = _previousVolume; _previousVolume = 0; } else { _previousVolume = _currentVolume; _currentVolume = 0; } SetVolume(_currentVolume); } public void ClearQueue() => SendCommand(new ClearCommand()); public void PlayCommand() => SendCommand(new PlayCommand(0)); public void AddSong(string Uri) { SendCommand(new AddCommand(Uri)); } public void ClearAddAndPlay(string Uri) { CommandList commandList = new CommandList(new IMpcCommand[] { new ClearCommand(), new AddCommand(Uri), new PlayCommand(0) }); SendCommand(commandList); } public async Task QueryPlaylist() => _Playlist = await SafelySendCommandAsync(new PlaylistCommand()); public int GetPlaylistCount() { if (_Playlist == null) return 0; return _Playlist.ToArray().Length; } public void UpdateDB() => SendCommand(new UpdateCommand()); private static string FormatTime(TimeSpan time) { string FormattedTime = ""; if (time.Days == 1) FormattedTime += $"{time.Days} {Resources.Resources.Day}, "; else if (time.Days > 1) FormattedTime += $"{time.Days} {Resources.Resources.Days}, "; if (time.Hours == 1) FormattedTime += $"{time.Hours} {Resources.Resources.Hour}, "; else FormattedTime += $"{time.Hours} {Resources.Resources.Hours}, "; if (time.Minutes == 1) FormattedTime += $"{time.Minutes} {Resources.Resources.Minute}, "; else FormattedTime += $"{time.Minutes} {Resources.Resources.Minutes}, "; if (time.Seconds == 1) FormattedTime += $"{time.Seconds} {Resources.Resources.Second}"; else FormattedTime += $"{time.Seconds} {Resources.Resources.Seconds}"; return FormattedTime; } public async void QueryStats() { Dictionary Response = await SafelySendCommandAsync(new StatsCommand()); if (Response == null) return; _stats.Songs = int.Parse(Response["songs"]); _stats.Albums = int.Parse(Response["albums"]); _stats.Artists = int.Parse(Response["artists"]); _stats.Uptime = FormatTime(TimeSpan.FromSeconds(int.Parse(Response["uptime"]))); _stats.TotalPlaytime = FormatTime(TimeSpan.FromSeconds(int.Parse(Response["db_playtime"]))); _stats.TotalTimePlayed = FormatTime(TimeSpan.FromSeconds(int.Parse(Response["playtime"]))); DateTime date = new DateTime(1970, 1, 1).AddSeconds(int.Parse(Response["db_update"])).ToLocalTime(); string dayOfWeek = Resources.Resources.Culture.DateTimeFormat.GetDayName(date.DayOfWeek); _stats.DatabaseUpdate = dayOfWeek + " " + date.ToString("dd/MM/yyyy @ HH:mm"); } } }