ChannelPostScanTask.cs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. using MediaBrowser.Common.Progress;
  2. using MediaBrowser.Controller.Channels;
  3. using MediaBrowser.Controller.Library;
  4. using MediaBrowser.Model.Channels;
  5. using MediaBrowser.Model.Logging;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Threading;
  10. using System.Threading.Tasks;
  11. namespace MediaBrowser.Server.Implementations.Channels
  12. {
  13. public class ChannelPostScanTask : ILibraryPostScanTask
  14. {
  15. private readonly IChannelManager _channelManager;
  16. private readonly IUserManager _userManager;
  17. private readonly ILogger _logger;
  18. public ChannelPostScanTask(IChannelManager channelManager, IUserManager userManager, ILogger logger)
  19. {
  20. _channelManager = channelManager;
  21. _userManager = userManager;
  22. _logger = logger;
  23. }
  24. public async Task Run(IProgress<double> progress, CancellationToken cancellationToken)
  25. {
  26. var users = _userManager.Users
  27. .Select(i => i.Id.ToString("N"))
  28. .ToList();
  29. var numComplete = 0;
  30. foreach (var user in users)
  31. {
  32. double percentPerUser = 1;
  33. percentPerUser /= users.Count;
  34. var startingPercent = numComplete * percentPerUser * 100;
  35. var innerProgress = new ActionableProgress<double>();
  36. innerProgress.RegisterAction(p => progress.Report(startingPercent + (percentPerUser * p)));
  37. await DownloadContent(user, cancellationToken, innerProgress).ConfigureAwait(false);
  38. numComplete++;
  39. double percent = numComplete;
  40. percent /= users.Count;
  41. progress.Report(percent * 100);
  42. }
  43. progress.Report(100);
  44. }
  45. private async Task DownloadContent(string user, CancellationToken cancellationToken, IProgress<double> progress)
  46. {
  47. var channels = await _channelManager.GetChannelsInternal(new ChannelQuery
  48. {
  49. UserId = user
  50. }, cancellationToken);
  51. var numComplete = 0;
  52. var numItems = channels.Items.Length;
  53. foreach (var channel in channels.Items)
  54. {
  55. var channelId = channel.Id.ToString("N");
  56. var features = _channelManager.GetChannelFeatures(channelId);
  57. const int currentRefreshLevel = 1;
  58. var maxRefreshLevel = features.AutoRefreshLevels ?? 1;
  59. var innerProgress = new ActionableProgress<double>();
  60. var startingNumberComplete = numComplete;
  61. innerProgress.RegisterAction(p =>
  62. {
  63. double innerPercent = startingNumberComplete;
  64. innerPercent += (p / 100);
  65. innerPercent /= numItems;
  66. progress.Report(innerPercent * 100);
  67. });
  68. try
  69. {
  70. await GetAllItems(user, channelId, null, currentRefreshLevel, maxRefreshLevel, innerProgress, cancellationToken).ConfigureAwait(false);
  71. }
  72. catch (Exception ex)
  73. {
  74. _logger.ErrorException("Error getting channel content", ex);
  75. }
  76. numComplete++;
  77. double percent = numComplete;
  78. percent /= numItems;
  79. progress.Report(percent * 100);
  80. }
  81. progress.Report(100);
  82. }
  83. private async Task GetAllItems(string user, string channelId, string folderId, int currentRefreshLevel, int maxRefreshLevel, IProgress<double> progress, CancellationToken cancellationToken)
  84. {
  85. var folderItems = new List<string>();
  86. var innerProgress = new ActionableProgress<double>();
  87. innerProgress.RegisterAction(p => progress.Report(p / 2));
  88. var result = await _channelManager.GetChannelItemsInternal(new ChannelItemQuery
  89. {
  90. ChannelId = channelId,
  91. UserId = user,
  92. FolderId = folderId
  93. }, innerProgress, cancellationToken);
  94. folderItems.AddRange(result.Items.Where(i => i.IsFolder).Select(i => i.Id.ToString("N")));
  95. var totalRetrieved = result.Items.Length;
  96. var totalCount = result.TotalRecordCount;
  97. while (totalRetrieved < totalCount)
  98. {
  99. result = await _channelManager.GetChannelItemsInternal(new ChannelItemQuery
  100. {
  101. ChannelId = channelId,
  102. UserId = user,
  103. StartIndex = totalRetrieved,
  104. FolderId = folderId
  105. }, new Progress<double>(), cancellationToken);
  106. folderItems.AddRange(result.Items.Where(i => i.IsFolder).Select(i => i.Id.ToString("N")));
  107. totalRetrieved += result.Items.Length;
  108. totalCount = result.TotalRecordCount;
  109. }
  110. progress.Report(50);
  111. if (currentRefreshLevel < maxRefreshLevel)
  112. {
  113. var numComplete = 0;
  114. var numItems = folderItems.Count;
  115. foreach (var folder in folderItems)
  116. {
  117. try
  118. {
  119. innerProgress = new ActionableProgress<double>();
  120. var startingNumberComplete = numComplete;
  121. innerProgress.RegisterAction(p =>
  122. {
  123. double innerPercent = startingNumberComplete;
  124. innerPercent += (p / 100);
  125. innerPercent /= numItems;
  126. progress.Report((innerPercent * 50) + 50);
  127. });
  128. await GetAllItems(user, channelId, folder, currentRefreshLevel + 1, maxRefreshLevel, innerProgress, cancellationToken).ConfigureAwait(false);
  129. }
  130. catch (Exception ex)
  131. {
  132. _logger.ErrorException("Error getting channel content", ex);
  133. }
  134. numComplete++;
  135. double percent = numComplete;
  136. percent /= numItems;
  137. progress.Report((percent * 50) + 50);
  138. }
  139. }
  140. progress.Report(100);
  141. }
  142. }
  143. }