如果Hangfire已经在运行,请停止启动定期作业

时间:2016-06-02 10:29:20

标签: asp.net-mvc concurrency attributes hangfire

我在Hangfire中设置了一系列定期作业,所有作业间隔为20分钟。

他们都调用相同的方法,但参数不同。 e.g。

职位ID test_1 => MyTestMethod(1)
职位ID test_50 => MyTestMethod(50)
职位ID test_37 => MyTestMethod(37)

有时,作业可能需要比其间隔更长的时间,即超过20分钟,我想确保作业不会再次运行,而另一个实例已经在运行。

此处DisableConcurrentExecutionAttribute不合适,因为该方法可以同时运行,而不是使用相同的参数。

我尝试使用静态Dictionary<string, DateTime>跟踪作业是否已在运行,以便它可以阻止并发方法调用,但问题是如果应用程序因任何原因重新启动,那么它就会#34 ;忘记&#34;当前正在运行的工作(当然Hangfire将继续在后台运行)

更新
我还尝试将 JobState 表添加到我的数据库以跟踪正在运行的作业,然后在MyTestMethod启动时检查该表,但这依赖于MyTestMethod设置 running = 1 当它结束并且线程在该作业的中途(无论出于何种原因)中途崩溃时,它会启动并且 running = 0 ,从而阻止作业再次运行。

我确信我可以通过简单地按作业ID查询Hangfire工作状态来解决这个问题 - 我只是无法找到如何做到这一点?

2 个答案:

答案 0 :(得分:6)

我有类似的要求,基本上我想使用DisableConcurrentExecutionAttribute,但要考虑参数。这样,如果一个作业使用相同的参数排队,它仍然会不会并行运行。我拿了示例DisableMultipleQueuedItemsFilter,它实际上删除了作业并修改了DisableConcurrentExecutionAttribute以使用参数。不同之处在于,如果作业具有相同的参数列表并且并行运行,那么作业将排队。

这里可以看到一个完整的示例,其中包含两个属性:https://gist.github.com/sbosell/3831f5bb893b20e82c72467baf8aefea

属性的相关代码:

 public class DisableConcurrentExecutionWithParametersAttribute : JobFilterAttribute, IServerFilter
    {
        private readonly int _timeoutInSeconds;

        public DisableConcurrentExecutionWithParametersAttribute (int timeoutInSeconds)
        {
            if (timeoutInSeconds < 0) throw new ArgumentException("Timeout argument value should be greater that zero.");

            _timeoutInSeconds = timeoutInSeconds;
        }

        public void OnPerforming(PerformingContext filterContext)
        {
            var resource = GetResource(filterContext.BackgroundJob.Job);

            var timeout = TimeSpan.FromSeconds(_timeoutInSeconds);

            var distributedLock = filterContext.Connection.AcquireDistributedLock(resource, timeout);
            filterContext.Items["DistributedLock"] = distributedLock;
        }

        public void OnPerformed(PerformedContext filterContext)
        {
            if (!filterContext.Items.ContainsKey("DistributedLock"))
            {
                throw new InvalidOperationException("Can not release a distributed lock: it was not acquired.");
            }

            var distributedLock = (IDisposable)filterContext.Items["DistributedLock"];
            distributedLock.Dispose();
        }

        private static string GetFingerprint(Job job)
        {
            var parameters = string.Empty;
            if (job?.Arguments != null)
            {
                parameters = string.Join(".", job.Arguments);
            }
            if (job?.Type == null || job.Method == null)
            {
                return string.Empty;
            }
            var payload = $"{job.Type.FullName}.{job.Method.Name}.{parameters}";
            var hash = SHA256.Create().ComputeHash(System.Text.Encoding.UTF8.GetBytes(payload));
            var fingerprint = Convert.ToBase64String(hash);
            return fingerprint;
        }
        private static string GetResource(Job job)
        {
            return GetFingerprint(job);
        }
    }

答案 1 :(得分:4)

您可以使用一个属性来阻止并发执行(一段时间):

[DisableConcurrentExecution(3600)] // argument in seconds, e.g., an hour
public void MyTestMethod(int id) {
  // ...
}