jellyfin/Emby.Server.Implementations/IO/FileRefresher.cs

232 lines
7.0 KiB
C#
Raw Normal View History

using System;
2016-05-25 21:11:27 -07:00
using System.Collections.Generic;
using System.IO;
using System.Linq;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
2016-10-21 19:08:34 -07:00
using MediaBrowser.Model.Extensions;
using MediaBrowser.Model.IO;
2016-11-04 12:51:59 -07:00
using MediaBrowser.Model.System;
2016-10-23 12:47:34 -07:00
using MediaBrowser.Model.Tasks;
using MediaBrowser.Model.Threading;
using Microsoft.Extensions.Logging;
2016-05-25 21:11:27 -07:00
2016-11-04 11:56:47 -07:00
namespace Emby.Server.Implementations.IO
2016-05-25 21:11:27 -07:00
{
public class FileRefresher : IDisposable
{
private ILogger Logger { get; set; }
private ITaskManager TaskManager { get; set; }
private ILibraryManager LibraryManager { get; set; }
private IServerConfigurationManager ConfigurationManager { get; set; }
private readonly IFileSystem _fileSystem;
private readonly List<string> _affectedPaths = new List<string>();
private ITimer _timer;
private readonly ITimerFactory _timerFactory;
2016-05-25 21:11:27 -07:00
private readonly object _timerLock = new object();
public string Path { get; private set; }
public event EventHandler<EventArgs> Completed;
2016-11-04 12:51:59 -07:00
private readonly IEnvironmentInfo _environmentInfo;
2017-05-11 22:00:45 -07:00
private readonly ILibraryManager _libraryManager;
2016-05-25 21:11:27 -07:00
2017-05-11 22:00:45 -07:00
public FileRefresher(string path, IFileSystem fileSystem, IServerConfigurationManager configurationManager, ILibraryManager libraryManager, ITaskManager taskManager, ILogger logger, ITimerFactory timerFactory, IEnvironmentInfo environmentInfo, ILibraryManager libraryManager1)
2016-05-25 21:11:27 -07:00
{
logger.LogDebug("New file refresher created for {0}", path);
Path = path;
2016-05-25 21:11:27 -07:00
_fileSystem = fileSystem;
ConfigurationManager = configurationManager;
LibraryManager = libraryManager;
TaskManager = taskManager;
Logger = logger;
_timerFactory = timerFactory;
2016-11-04 12:51:59 -07:00
_environmentInfo = environmentInfo;
2017-05-11 22:00:45 -07:00
_libraryManager = libraryManager1;
2016-07-02 19:47:39 -07:00
AddPath(path);
2016-05-25 21:11:27 -07:00
}
private void AddAffectedPath(string path)
{
2018-09-12 10:26:21 -07:00
if (string.IsNullOrEmpty(path))
{
throw new ArgumentNullException(nameof(path));
}
if (!_affectedPaths.Contains(path, StringComparer.Ordinal))
{
_affectedPaths.Add(path);
}
}
public void AddPath(string path)
{
2018-09-12 10:26:21 -07:00
if (string.IsNullOrEmpty(path))
{
throw new ArgumentNullException(nameof(path));
}
lock (_timerLock)
{
AddAffectedPath(path);
}
RestartTimer();
}
public void RestartTimer()
2016-05-25 21:11:27 -07:00
{
2016-08-10 20:56:01 -07:00
if (_disposed)
{
return;
}
2016-05-25 21:11:27 -07:00
lock (_timerLock)
{
2016-09-16 23:08:38 -07:00
if (_disposed)
{
return;
}
2016-05-25 21:11:27 -07:00
if (_timer == null)
{
_timer = _timerFactory.Create(OnTimerCallback, null, TimeSpan.FromSeconds(ConfigurationManager.Configuration.LibraryMonitorDelay), TimeSpan.FromMilliseconds(-1));
2016-05-25 21:11:27 -07:00
}
else
{
_timer.Change(TimeSpan.FromSeconds(ConfigurationManager.Configuration.LibraryMonitorDelay), TimeSpan.FromMilliseconds(-1));
}
}
}
public void ResetPath(string path, string affectedFile)
{
lock (_timerLock)
{
Logger.LogDebug("Resetting file refresher from {0} to {1}", Path, path);
Path = path;
AddAffectedPath(path);
2018-09-12 10:26:21 -07:00
if (!string.IsNullOrEmpty(affectedFile))
{
AddAffectedPath(affectedFile);
}
}
RestartTimer();
}
2017-08-16 10:30:16 -07:00
private void OnTimerCallback(object state)
2016-05-25 21:11:27 -07:00
{
2016-06-11 08:56:15 -07:00
List<string> paths;
lock (_timerLock)
{
paths = _affectedPaths.ToList();
}
Logger.LogDebug("Timer stopped.");
2016-05-25 21:11:27 -07:00
DisposeTimer();
Completed?.Invoke(this, EventArgs.Empty);
2016-05-25 21:11:27 -07:00
try
{
2017-08-16 10:30:16 -07:00
ProcessPathChanges(paths.ToList());
2016-05-25 21:11:27 -07:00
}
catch (Exception ex)
{
2018-12-20 05:11:26 -07:00
Logger.LogError(ex, "Error processing directory changes");
2016-05-25 21:11:27 -07:00
}
}
2017-08-16 10:30:16 -07:00
private void ProcessPathChanges(List<string> paths)
2016-05-25 21:11:27 -07:00
{
var itemsToRefresh = paths
2016-09-20 12:43:27 -07:00
.Distinct(StringComparer.OrdinalIgnoreCase)
2016-05-25 21:11:27 -07:00
.Select(GetAffectedBaseItem)
.Where(item => item != null)
2016-09-20 12:43:27 -07:00
.DistinctBy(i => i.Id)
2016-05-25 21:11:27 -07:00
.ToList();
foreach (var item in itemsToRefresh)
{
2017-10-03 11:39:37 -07:00
if (item is AggregateFolder)
{
continue;
}
Logger.LogInformation("{name} ({path}) will be refreshed.", item.Name, item.Path);
2016-05-25 21:11:27 -07:00
try
{
2017-05-27 00:19:09 -07:00
item.ChangedExternally();
2016-05-25 21:11:27 -07:00
}
catch (IOException ex)
{
2019-01-07 16:27:46 -07:00
// For now swallow and log.
2016-05-25 21:11:27 -07:00
// Research item: If an IOException occurs, the item may be in a disconnected state (media unavailable)
// Should we remove it from it's parent?
2018-12-20 05:11:26 -07:00
Logger.LogError(ex, "Error refreshing {name}", item.Name);
2016-05-25 21:11:27 -07:00
}
catch (Exception ex)
{
2018-12-20 05:11:26 -07:00
Logger.LogError(ex, "Error refreshing {name}", item.Name);
2016-05-25 21:11:27 -07:00
}
}
}
/// <summary>
/// Gets the affected base item.
/// </summary>
/// <param name="path">The path.</param>
/// <returns>BaseItem.</returns>
private BaseItem GetAffectedBaseItem(string path)
{
BaseItem item = null;
while (item == null && !string.IsNullOrEmpty(path))
{
item = LibraryManager.FindByPath(path, null);
path = System.IO.Path.GetDirectoryName(path);
2016-05-25 21:11:27 -07:00
}
if (item != null)
{
// If the item has been deleted find the first valid parent that still exists
while (!_fileSystem.DirectoryExists(item.Path) && !_fileSystem.FileExists(item.Path))
{
2018-09-12 10:26:21 -07:00
item = item.GetOwner() ?? item.GetParent();
2016-05-25 21:11:27 -07:00
if (item == null)
{
break;
}
}
}
return item;
}
private void DisposeTimer()
2016-05-25 21:11:27 -07:00
{
lock (_timerLock)
{
if (_timer != null)
{
_timer.Dispose();
2016-09-16 23:08:38 -07:00
_timer = null;
2016-05-25 21:11:27 -07:00
}
}
}
2016-08-10 20:56:01 -07:00
private bool _disposed;
2016-05-25 21:11:27 -07:00
public void Dispose()
{
2016-08-10 20:56:01 -07:00
_disposed = true;
2016-05-25 21:11:27 -07:00
DisposeTimer();
}
}
}