mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 16:22:28 -04:00 
			
		
		
		
	* Moved Collapse Series with relationships into a user preference rather than library setting. * Fixed bookmarks not converting to webp after initial save * Fixed a bug where when merging we'd print out a duplicate series error when we shouldn't have * Fixed a bug where clicking on a genre or tag from server stats wouldn't load all-series page in a filtered state. * Implemented the ability to have Login role and thus disable accounts. * Ensure first time flow gets the Login role * Refactored user management screen so that pending users can be edited or deleted before the end user accepts the invite. A side effect is old legacy users that were here before email was required can now be deleted. * Show a progress bar under the main series image on larger viewports to show whole series progress. * Removed code no longer needed * Cleanup tags, people, collections without connections after editing series metadata. * Moved the Entity Builders to the main project
		
			
				
	
	
		
			430 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			430 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Collections.Immutable;
 | |
| using System.Linq;
 | |
| using System.Threading.Tasks;
 | |
| using API.Data;
 | |
| using API.Entities.Enums;
 | |
| using API.Helpers.Converters;
 | |
| using API.Services.Tasks;
 | |
| using API.Services.Tasks.Metadata;
 | |
| using Hangfire;
 | |
| using Microsoft.Extensions.Logging;
 | |
| 
 | |
| namespace API.Services;
 | |
| 
 | |
| public interface ITaskScheduler
 | |
| {
 | |
|     Task ScheduleTasks();
 | |
|     Task ScheduleStatsTasks();
 | |
|     void ScheduleUpdaterTasks();
 | |
|     void ScanFolder(string folderPath, TimeSpan delay);
 | |
|     void ScanFolder(string folderPath);
 | |
|     void ScanLibrary(int libraryId, bool force = false);
 | |
|     void CleanupChapters(int[] chapterIds);
 | |
|     void RefreshMetadata(int libraryId, bool forceUpdate = true);
 | |
|     void RefreshSeriesMetadata(int libraryId, int seriesId, bool forceUpdate = false);
 | |
|     void ScanSeries(int libraryId, int seriesId, bool forceUpdate = false);
 | |
|     void AnalyzeFilesForSeries(int libraryId, int seriesId, bool forceUpdate = false);
 | |
|     void AnalyzeFilesForLibrary(int libraryId, bool forceUpdate = false);
 | |
|     void CancelStatsTasks();
 | |
|     Task RunStatCollection();
 | |
|     void ScanSiteThemes();
 | |
|     Task CovertAllCoversToWebP();
 | |
|     Task CleanupDbEntries();
 | |
| }
 | |
| public class TaskScheduler : ITaskScheduler
 | |
| {
 | |
|     private readonly ICacheService _cacheService;
 | |
|     private readonly ILogger<TaskScheduler> _logger;
 | |
|     private readonly IScannerService _scannerService;
 | |
|     private readonly IUnitOfWork _unitOfWork;
 | |
|     private readonly IMetadataService _metadataService;
 | |
|     private readonly IBackupService _backupService;
 | |
|     private readonly ICleanupService _cleanupService;
 | |
| 
 | |
|     private readonly IStatsService _statsService;
 | |
|     private readonly IVersionUpdaterService _versionUpdaterService;
 | |
|     private readonly IThemeService _themeService;
 | |
|     private readonly IWordCountAnalyzerService _wordCountAnalyzerService;
 | |
|     private readonly IStatisticService _statisticService;
 | |
|     private readonly IBookmarkService _bookmarkService;
 | |
| 
 | |
|     public static BackgroundJobServer Client => new BackgroundJobServer();
 | |
|     public const string ScanQueue = "scan";
 | |
|     public const string DefaultQueue = "default";
 | |
|     public const string RemoveFromWantToReadTaskId = "remove-from-want-to-read";
 | |
|     public const string UpdateYearlyStatsTaskId = "update-yearly-stats";
 | |
|     public const string CleanupDbTaskId = "cleanup-db";
 | |
|     public const string CleanupTaskId = "cleanup";
 | |
|     public const string BackupTaskId = "backup";
 | |
|     public const string ScanLibrariesTaskId = "scan-libraries";
 | |
|     public const string ReportStatsTaskId = "report-stats";
 | |
| 
 | |
|     private static readonly ImmutableArray<string> ScanTasks = ImmutableArray.Create("ScannerService", "ScanLibrary", "ScanLibraries", "ScanFolder", "ScanSeries");
 | |
| 
 | |
|     private static readonly Random Rnd = new Random();
 | |
| 
 | |
| 
 | |
|     public TaskScheduler(ICacheService cacheService, ILogger<TaskScheduler> logger, IScannerService scannerService,
 | |
|         IUnitOfWork unitOfWork, IMetadataService metadataService, IBackupService backupService,
 | |
|         ICleanupService cleanupService, IStatsService statsService, IVersionUpdaterService versionUpdaterService,
 | |
|         IThemeService themeService, IWordCountAnalyzerService wordCountAnalyzerService, IStatisticService statisticService,
 | |
|         IBookmarkService bookmarkService)
 | |
|     {
 | |
|         _cacheService = cacheService;
 | |
|         _logger = logger;
 | |
|         _scannerService = scannerService;
 | |
|         _unitOfWork = unitOfWork;
 | |
|         _metadataService = metadataService;
 | |
|         _backupService = backupService;
 | |
|         _cleanupService = cleanupService;
 | |
|         _statsService = statsService;
 | |
|         _versionUpdaterService = versionUpdaterService;
 | |
|         _themeService = themeService;
 | |
|         _wordCountAnalyzerService = wordCountAnalyzerService;
 | |
|         _statisticService = statisticService;
 | |
|         _bookmarkService = bookmarkService;
 | |
|     }
 | |
| 
 | |
|     public async Task ScheduleTasks()
 | |
|     {
 | |
|         _logger.LogInformation("Scheduling reoccurring tasks");
 | |
| 
 | |
|         var setting = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.TaskScan)).Value;
 | |
|         if (setting != null)
 | |
|         {
 | |
|             var scanLibrarySetting = setting;
 | |
|             _logger.LogDebug("Scheduling Scan Library Task for {Setting}", scanLibrarySetting);
 | |
|             RecurringJob.AddOrUpdate(ScanLibrariesTaskId, () => _scannerService.ScanLibraries(),
 | |
|                 () => CronConverter.ConvertToCronNotation(scanLibrarySetting), TimeZoneInfo.Local);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             RecurringJob.AddOrUpdate(ScanLibrariesTaskId, () => ScanLibraries(), Cron.Daily, TimeZoneInfo.Local);
 | |
|         }
 | |
| 
 | |
|         setting = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.TaskBackup)).Value;
 | |
|         if (setting != null)
 | |
|         {
 | |
|             _logger.LogDebug("Scheduling Backup Task for {Setting}", setting);
 | |
|             RecurringJob.AddOrUpdate(BackupTaskId, () => _backupService.BackupDatabase(), () => CronConverter.ConvertToCronNotation(setting), TimeZoneInfo.Local);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             RecurringJob.AddOrUpdate(BackupTaskId, () => _backupService.BackupDatabase(), Cron.Weekly, TimeZoneInfo.Local);
 | |
|         }
 | |
| 
 | |
|         RecurringJob.AddOrUpdate(CleanupTaskId, () => _cleanupService.Cleanup(), Cron.Daily, TimeZoneInfo.Local);
 | |
|         RecurringJob.AddOrUpdate(CleanupDbTaskId, () => _cleanupService.CleanupDbEntries(), Cron.Daily, TimeZoneInfo.Local);
 | |
|         RecurringJob.AddOrUpdate(RemoveFromWantToReadTaskId, () => _cleanupService.CleanupWantToRead(), Cron.Daily, TimeZoneInfo.Local);
 | |
|         RecurringJob.AddOrUpdate(UpdateYearlyStatsTaskId, () => _statisticService.UpdateServerStatistics(), Cron.Monthly, TimeZoneInfo.Local);
 | |
|     }
 | |
| 
 | |
|     #region StatsTasks
 | |
| 
 | |
| 
 | |
|     public async Task ScheduleStatsTasks()
 | |
|     {
 | |
|         var allowStatCollection  = (await _unitOfWork.SettingsRepository.GetSettingsDtoAsync()).AllowStatCollection;
 | |
|         if (!allowStatCollection)
 | |
|         {
 | |
|             _logger.LogDebug("User has opted out of stat collection, not registering tasks");
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogDebug("Scheduling stat collection daily");
 | |
|         RecurringJob.AddOrUpdate(ReportStatsTaskId, () => _statsService.Send(), Cron.Daily(Rnd.Next(0, 22)), TimeZoneInfo.Local);
 | |
|     }
 | |
| 
 | |
|     public void AnalyzeFilesForLibrary(int libraryId, bool forceUpdate = false)
 | |
|     {
 | |
|         BackgroundJob.Enqueue(() => _wordCountAnalyzerService.ScanLibrary(libraryId, forceUpdate));
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Upon cancelling stat, we do report to the Stat service that we are no longer going to be reporting
 | |
|     /// </summary>
 | |
|     public void CancelStatsTasks()
 | |
|     {
 | |
|         _logger.LogDebug("Stopping Stat collection as user has opted out");
 | |
|         RecurringJob.RemoveIfExists(ReportStatsTaskId);
 | |
|         _statsService.SendCancellation();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// First time run stat collection. Executes immediately on a background thread. Does not block.
 | |
|     /// </summary>
 | |
|     /// <remarks>Schedules it for 1 day in the future to ensure we don't have users that try the software out</remarks>
 | |
|     public async Task RunStatCollection()
 | |
|     {
 | |
|         var allowStatCollection  = (await _unitOfWork.SettingsRepository.GetSettingsDtoAsync()).AllowStatCollection;
 | |
|         if (!allowStatCollection)
 | |
|         {
 | |
|             _logger.LogDebug("User has opted out of stat collection, not sending stats");
 | |
|             return;
 | |
|         }
 | |
|         BackgroundJob.Schedule(() => _statsService.Send(), DateTimeOffset.Now.AddDays(1));
 | |
|     }
 | |
| 
 | |
|     public void ScanSiteThemes()
 | |
|     {
 | |
|         if (HasAlreadyEnqueuedTask("ThemeService", "Scan", Array.Empty<object>(), ScanQueue))
 | |
|         {
 | |
|             _logger.LogInformation("A Theme Scan is already running");
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Enqueueing Site Theme scan");
 | |
|         BackgroundJob.Enqueue(() => _themeService.Scan());
 | |
|     }
 | |
| 
 | |
|     public async Task CovertAllCoversToWebP()
 | |
|     {
 | |
|         await _bookmarkService.ConvertAllCoverToWebP();
 | |
|         _logger.LogInformation("[BookmarkService] Queuing tasks to update Series and Volume references via Cover Refresh");
 | |
|         var libraryIds = await _unitOfWork.LibraryRepository.GetLibrariesAsync();
 | |
|         foreach (var lib in libraryIds)
 | |
|         {
 | |
|             RefreshMetadata(lib.Id, false);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     #region UpdateTasks
 | |
| 
 | |
|     public void ScheduleUpdaterTasks()
 | |
|     {
 | |
|         _logger.LogInformation("Scheduling Auto-Update tasks");
 | |
|         // Schedule update check between noon and 6pm local time
 | |
|         RecurringJob.AddOrUpdate("check-updates", () => CheckForUpdate(), Cron.Daily(Rnd.Next(12, 18)), TimeZoneInfo.Local);
 | |
|     }
 | |
| 
 | |
|     public void ScanFolder(string folderPath, TimeSpan delay)
 | |
|     {
 | |
|         var normalizedFolder = Tasks.Scanner.Parser.Parser.NormalizePath(folderPath);
 | |
|         if (HasAlreadyEnqueuedTask(ScannerService.Name, "ScanFolder", new object[] { normalizedFolder }))
 | |
|         {
 | |
|             _logger.LogInformation("Skipped scheduling ScanFolder for {Folder} as a job already queued",
 | |
|                 normalizedFolder);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Scheduling ScanFolder for {Folder}", normalizedFolder);
 | |
|         BackgroundJob.Schedule(() => _scannerService.ScanFolder(normalizedFolder), delay);
 | |
|     }
 | |
| 
 | |
|     public void ScanFolder(string folderPath)
 | |
|     {
 | |
|         var normalizedFolder = Tasks.Scanner.Parser.Parser.NormalizePath(folderPath);
 | |
|         if (HasAlreadyEnqueuedTask(ScannerService.Name, "ScanFolder", new object[] {normalizedFolder}))
 | |
|         {
 | |
|             _logger.LogInformation("Skipped scheduling ScanFolder for {Folder} as a job already queued",
 | |
|                 normalizedFolder);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Scheduling ScanFolder for {Folder}", normalizedFolder);
 | |
|         _scannerService.ScanFolder(normalizedFolder);
 | |
|     }
 | |
| 
 | |
|     #endregion
 | |
| 
 | |
|     public async Task CleanupDbEntries()
 | |
|     {
 | |
|         await _cleanupService.CleanupDbEntries();
 | |
|     }
 | |
| 
 | |
|     public void ScanLibraries()
 | |
|     {
 | |
|         if (RunningAnyTasksByMethod(ScanTasks, ScanQueue))
 | |
|         {
 | |
|             _logger.LogInformation("A Scan is already running, rescheduling ScanLibraries in 3 hours");
 | |
|             BackgroundJob.Schedule(() => ScanLibraries(), TimeSpan.FromHours(3));
 | |
|             return;
 | |
|         }
 | |
|         _scannerService.ScanLibraries();
 | |
|     }
 | |
| 
 | |
|     public void ScanLibrary(int libraryId, bool force = false)
 | |
|     {
 | |
|         if (HasScanTaskRunningForLibrary(libraryId))
 | |
|         {
 | |
|             _logger.LogInformation("A duplicate request for Library Scan on library {LibraryId} occured. Skipping", libraryId);
 | |
|             return;
 | |
|         }
 | |
|         if (RunningAnyTasksByMethod(ScanTasks, ScanQueue))
 | |
|         {
 | |
|             _logger.LogInformation("A Library Scan is already running, rescheduling ScanLibrary in 3 hours");
 | |
|             BackgroundJob.Schedule(() => ScanLibrary(libraryId, force), TimeSpan.FromHours(3));
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Enqueuing library scan for: {LibraryId}", libraryId);
 | |
|         BackgroundJob.Enqueue(() => _scannerService.ScanLibrary(libraryId, force));
 | |
|         // When we do a scan, force cache to re-unpack in case page numbers change
 | |
|         BackgroundJob.Enqueue(() => _cleanupService.CleanupCacheAndTempDirectories());
 | |
|     }
 | |
| 
 | |
|     public void CleanupChapters(int[] chapterIds)
 | |
|     {
 | |
|         BackgroundJob.Enqueue(() => _cacheService.CleanupChapters(chapterIds));
 | |
|     }
 | |
| 
 | |
|     public void RefreshMetadata(int libraryId, bool forceUpdate = true)
 | |
|     {
 | |
|         var alreadyEnqueued = HasAlreadyEnqueuedTask(MetadataService.Name, "GenerateCoversForLibrary",
 | |
|                                   new object[] {libraryId, true}) ||
 | |
|                               HasAlreadyEnqueuedTask("MetadataService", "GenerateCoversForLibrary",
 | |
|                                   new object[] {libraryId, false});
 | |
|         if (alreadyEnqueued)
 | |
|         {
 | |
|             _logger.LogInformation("A duplicate request to refresh metadata for library occured. Skipping");
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Enqueuing library metadata refresh for: {LibraryId}", libraryId);
 | |
|         BackgroundJob.Enqueue(() => _metadataService.GenerateCoversForLibrary(libraryId, forceUpdate));
 | |
|     }
 | |
| 
 | |
|     public void RefreshSeriesMetadata(int libraryId, int seriesId, bool forceUpdate = false)
 | |
|     {
 | |
|         if (HasAlreadyEnqueuedTask(MetadataService.Name,"GenerateCoversForSeries",  new object[] {libraryId, seriesId, forceUpdate}))
 | |
|         {
 | |
|             _logger.LogInformation("A duplicate request to refresh metadata for library occured. Skipping");
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Enqueuing series metadata refresh for: {SeriesId}", seriesId);
 | |
|         BackgroundJob.Enqueue(() => _metadataService.GenerateCoversForSeries(libraryId, seriesId, forceUpdate));
 | |
|     }
 | |
| 
 | |
|     public void ScanSeries(int libraryId, int seriesId, bool forceUpdate = false)
 | |
|     {
 | |
|         if (HasAlreadyEnqueuedTask(ScannerService.Name, "ScanSeries", new object[] {seriesId, forceUpdate}, ScanQueue))
 | |
|         {
 | |
|             _logger.LogInformation("A duplicate request to scan series occured. Skipping");
 | |
|             return;
 | |
|         }
 | |
|         if (RunningAnyTasksByMethod(ScanTasks, ScanQueue))
 | |
|         {
 | |
|             _logger.LogInformation("A Scan is already running, rescheduling ScanSeries in 10 minutes");
 | |
|             BackgroundJob.Schedule(() => ScanSeries(libraryId, seriesId, forceUpdate), TimeSpan.FromMinutes(10));
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Enqueuing series scan for: {SeriesId}", seriesId);
 | |
|         BackgroundJob.Enqueue(() => _scannerService.ScanSeries(seriesId, forceUpdate));
 | |
|     }
 | |
| 
 | |
|     public void AnalyzeFilesForSeries(int libraryId, int seriesId, bool forceUpdate = false)
 | |
|     {
 | |
|         if (HasAlreadyEnqueuedTask("WordCountAnalyzerService", "ScanSeries", new object[] {libraryId, seriesId, forceUpdate}))
 | |
|         {
 | |
|             _logger.LogInformation("A duplicate request to scan series occured. Skipping");
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogInformation("Enqueuing analyze files scan for: {SeriesId}", seriesId);
 | |
|         BackgroundJob.Enqueue(() => _wordCountAnalyzerService.ScanSeries(libraryId, seriesId, forceUpdate));
 | |
|     }
 | |
| 
 | |
|     public void BackupDatabase()
 | |
|     {
 | |
|         BackgroundJob.Enqueue(() => _backupService.BackupDatabase());
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Not an external call. Only public so that we can call this for a Task
 | |
|     /// </summary>
 | |
|     // ReSharper disable once MemberCanBePrivate.Global
 | |
|     public async Task CheckForUpdate()
 | |
|     {
 | |
|         var update = await _versionUpdaterService.CheckForUpdate();
 | |
|         if (update == null) return;
 | |
|         await _versionUpdaterService.PushUpdate(update);
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// If there is an enqueued or scheduled task for <see cref="ScannerService.ScanLibrary"/> method
 | |
|     /// </summary>
 | |
|     /// <param name="libraryId"></param>
 | |
|     /// <param name="checkRunningJobs">Checks against jobs currently executing as well</param>
 | |
|     /// <returns></returns>
 | |
|     public static bool HasScanTaskRunningForLibrary(int libraryId, bool checkRunningJobs = true)
 | |
|     {
 | |
|         return
 | |
|             HasAlreadyEnqueuedTask(ScannerService.Name, "ScanLibrary", new object[] {libraryId, true}, ScanQueue, checkRunningJobs) ||
 | |
|             HasAlreadyEnqueuedTask(ScannerService.Name, "ScanLibrary", new object[] {libraryId, false}, ScanQueue, checkRunningJobs);
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// If there is an enqueued or scheduled task for <see cref="ScannerService.ScanSeries"/> method
 | |
|     /// </summary>
 | |
|     /// <param name="seriesId"></param>
 | |
|     /// <param name="checkRunningJobs">Checks against jobs currently executing as well</param>
 | |
|     /// <returns></returns>
 | |
|     public static bool HasScanTaskRunningForSeries(int seriesId, bool checkRunningJobs = true)
 | |
|     {
 | |
|         return
 | |
|             HasAlreadyEnqueuedTask(ScannerService.Name, "ScanSeries", new object[] {seriesId, true}, ScanQueue, checkRunningJobs) ||
 | |
|             HasAlreadyEnqueuedTask(ScannerService.Name, "ScanSeries", new object[] {seriesId, false}, ScanQueue, checkRunningJobs);
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Checks if this same invocation is already enqueued or scheduled
 | |
|     /// </summary>
 | |
|     /// <param name="methodName">Method name that was enqueued</param>
 | |
|     /// <param name="className">Class name the method resides on</param>
 | |
|     /// <param name="args">object[] of arguments in the order they are passed to enqueued job</param>
 | |
|     /// <param name="queue">Queue to check against. Defaults to "default"</param>
 | |
|     /// <param name="checkRunningJobs">Check against running jobs. Defaults to false.</param>
 | |
|     /// <returns></returns>
 | |
|     public static bool HasAlreadyEnqueuedTask(string className, string methodName, object[] args, string queue = DefaultQueue, bool checkRunningJobs = false)
 | |
|     {
 | |
|         var enqueuedJobs =  JobStorage.Current.GetMonitoringApi().EnqueuedJobs(queue, 0, int.MaxValue);
 | |
|         var ret = enqueuedJobs.Any(j => j.Value.InEnqueuedState &&
 | |
|                                      j.Value.Job.Method.DeclaringType != null && j.Value.Job.Args.SequenceEqual(args) &&
 | |
|                                      j.Value.Job.Method.Name.Equals(methodName) &&
 | |
|                                      j.Value.Job.Method.DeclaringType.Name.Equals(className));
 | |
|         if (ret) return true;
 | |
| 
 | |
|         var scheduledJobs = JobStorage.Current.GetMonitoringApi().ScheduledJobs(0, int.MaxValue);
 | |
|         ret = scheduledJobs.Any(j =>
 | |
|             j.Value.Job.Method.DeclaringType != null && j.Value.Job.Args.SequenceEqual(args) &&
 | |
|             j.Value.Job.Method.Name.Equals(methodName) &&
 | |
|             j.Value.Job.Method.DeclaringType.Name.Equals(className));
 | |
| 
 | |
|         if (ret) return true;
 | |
| 
 | |
|         if (checkRunningJobs)
 | |
|         {
 | |
|             var runningJobs = JobStorage.Current.GetMonitoringApi().ProcessingJobs(0, int.MaxValue);
 | |
|             return runningJobs.Any(j =>
 | |
|                 j.Value.Job.Method.DeclaringType != null && j.Value.Job.Args.SequenceEqual(args) &&
 | |
|                 j.Value.Job.Method.Name.Equals(methodName) &&
 | |
|                 j.Value.Job.Method.DeclaringType.Name.Equals(className));
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Checks against any jobs that are running or about to run
 | |
|     /// </summary>
 | |
|     /// <param name="classNames"></param>
 | |
|     /// <param name="queue"></param>
 | |
|     /// <returns></returns>
 | |
|     public static bool RunningAnyTasksByMethod(IEnumerable<string> classNames, string queue = DefaultQueue)
 | |
|     {
 | |
|         var enqueuedJobs =  JobStorage.Current.GetMonitoringApi().EnqueuedJobs(queue, 0, int.MaxValue);
 | |
|         var ret = enqueuedJobs.Any(j => !j.Value.InEnqueuedState &&
 | |
|                                      classNames.Contains(j.Value.Job.Method.DeclaringType?.Name));
 | |
|         if (ret) return true;
 | |
| 
 | |
|         var runningJobs = JobStorage.Current.GetMonitoringApi().ProcessingJobs(0, int.MaxValue);
 | |
|         return runningJobs.Any(j => classNames.Contains(j.Value.Job.Method.DeclaringType?.Name));
 | |
|     }
 | |
| }
 |