CleanDatabaseScheduledTask.cs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. using MediaBrowser.Common.Progress;
  2. using MediaBrowser.Controller.Entities;
  3. using MediaBrowser.Controller.Library;
  4. using MediaBrowser.Controller.Persistence;
  5. using MediaBrowser.Model.Entities;
  6. using MediaBrowser.Model.Logging;
  7. using System;
  8. using System.Collections.Generic;
  9. using System.Linq;
  10. using System.Threading;
  11. using System.Threading.Tasks;
  12. using MediaBrowser.Common.Configuration;
  13. using MediaBrowser.Model.IO;
  14. using MediaBrowser.Controller.Channels;
  15. using MediaBrowser.Controller.Entities.Audio;
  16. using MediaBrowser.Model.Tasks;
  17. namespace Emby.Server.Implementations.Data
  18. {
  19. public class CleanDatabaseScheduledTask : IScheduledTask
  20. {
  21. private readonly ILibraryManager _libraryManager;
  22. private readonly IItemRepository _itemRepo;
  23. private readonly ILogger _logger;
  24. private readonly IFileSystem _fileSystem;
  25. private readonly IApplicationPaths _appPaths;
  26. public CleanDatabaseScheduledTask(ILibraryManager libraryManager, IItemRepository itemRepo, ILogger logger, IFileSystem fileSystem, IApplicationPaths appPaths)
  27. {
  28. _libraryManager = libraryManager;
  29. _itemRepo = itemRepo;
  30. _logger = logger;
  31. _fileSystem = fileSystem;
  32. _appPaths = appPaths;
  33. }
  34. public string Name
  35. {
  36. get { return "Clean Database"; }
  37. }
  38. public string Description
  39. {
  40. get { return "Deletes obsolete content from the database."; }
  41. }
  42. public string Category
  43. {
  44. get { return "Library"; }
  45. }
  46. public async Task Execute(CancellationToken cancellationToken, IProgress<double> progress)
  47. {
  48. // Ensure these objects are lazy loaded.
  49. // Without this there is a deadlock that will need to be investigated
  50. var rootChildren = _libraryManager.RootFolder.Children.ToList();
  51. rootChildren = _libraryManager.GetUserRootFolder().Children.ToList();
  52. var innerProgress = new ActionableProgress<double>();
  53. innerProgress.RegisterAction(p =>
  54. {
  55. double newPercentCommplete = .45 * p;
  56. progress.Report(newPercentCommplete);
  57. });
  58. await CleanDeadItems(cancellationToken, innerProgress).ConfigureAwait(false);
  59. progress.Report(45);
  60. innerProgress = new ActionableProgress<double>();
  61. innerProgress.RegisterAction(p =>
  62. {
  63. double newPercentCommplete = 45 + .55 * p;
  64. progress.Report(newPercentCommplete);
  65. });
  66. await _itemRepo.UpdateInheritedValues(cancellationToken).ConfigureAwait(false);
  67. progress.Report(100);
  68. }
  69. private async Task CleanDeadItems(CancellationToken cancellationToken, IProgress<double> progress)
  70. {
  71. var itemIds = _libraryManager.GetItemIds(new InternalItemsQuery
  72. {
  73. HasDeadParentId = true
  74. });
  75. var numComplete = 0;
  76. var numItems = itemIds.Count;
  77. _logger.Debug("Cleaning {0} items with dead parent links", numItems);
  78. foreach (var itemId in itemIds)
  79. {
  80. cancellationToken.ThrowIfCancellationRequested();
  81. var item = _libraryManager.GetItemById(itemId);
  82. if (item != null)
  83. {
  84. _logger.Info("Cleaning item {0} type: {1} path: {2}", item.Name, item.GetType().Name, item.Path ?? string.Empty);
  85. await item.Delete(new DeleteOptions
  86. {
  87. DeleteFileLocation = false
  88. }).ConfigureAwait(false);
  89. }
  90. numComplete++;
  91. double percent = numComplete;
  92. percent /= numItems;
  93. progress.Report(percent * 100);
  94. }
  95. progress.Report(100);
  96. }
  97. /// <summary>
  98. /// Creates the triggers that define when the task will run
  99. /// </summary>
  100. /// <returns>IEnumerable{BaseTaskTrigger}.</returns>
  101. public IEnumerable<TaskTriggerInfo> GetDefaultTriggers()
  102. {
  103. return new[] {
  104. // Every so often
  105. new TaskTriggerInfo { Type = TaskTriggerInfo.TriggerInterval, IntervalTicks = TimeSpan.FromHours(24).Ticks}
  106. };
  107. }
  108. public string Key
  109. {
  110. get { return "CleanDatabase"; }
  111. }
  112. }
  113. }