Azure Service Bus,确定OnMessage是否停止处理

时间:2015-07-10 15:40:28

标签: c# azure azure-servicebus-queues

使用Azure ServiceBus和OnMessage调用我正在寻找一种方法来确定OnMessage事件泵是否停止从队列中读取。

我们与OnMessage的连接配置如下:

protected virtual void DoSubscription(string queueName, Func<QueueRequest, bool> callback)
{
    var client = GetClient(queueName, PollingTimeout);
    var transformCallback = new Action<BrokeredMessage>((message) =>
    {
        try
        {
            var request = message.ToQueueRequest();
            if (callback(request))
            {
                message.Complete();
            }
            else
            {
                message.Abandon();
                Log.Warn("DoSubscription: Message Failed to Process Gracefully: {0}{1}", Environment.NewLine, JsonConvert.SerializeObject(request));
            }
        }
        catch (Exception ex)
        {
            Log.Error("DoSubscription: Message Failed to Process With Exception:", ex);
            message.Abandon();
        }
    });
    var options = new OnMessageOptions
    {
        MaxConcurrentCalls = _config.GetInt("MaxThreadsPerQueue"),
        AutoComplete = false,
        AutoRenewTimeout = new TimeSpan(0,0,1)
    };
    options.ExceptionReceived += OnMessageError;
    client.OnMessage(transformCallback, options);
}

我们遇到的问题是在一段时间没有消息排队后,OnMessage事件泵无法获取排队的新消息,直到应用程序重新启动。

我意识到有很多方法可以使用Worker Roles来实现这一点,但是出于监控和管理的目的,我们决定在Web应用程序的应用程序启动中实现它。

1 个答案:

答案 0 :(得分:6)

因此,在与Microsoft的Azure支持团队通话后,当OnMessage或OnMessageAsync错误时,没有事件要捕获。由于这些不是阻塞调用,它启动事件泵并返回执行线程,这会产生一个挑战,以确定OnMessage *是否正在执行它的工作。

微软的建议是:

  • 创建自己的QueueClientBase类实现,该类公开您可以处理的OnClose,On *方法。但是,在执行此操作时,您必须自己处理邮件信封。
  • 在单独的线程循环中使用OnReceive,您可以自己捕获错误并立即重试。

然而,我确实探索了一些针对OnMessage的防弹,并发现了一些缓解我恐惧的事情。

  • OnMessage具有令人难以置信的容错能力
    • 我的笔记本电脑上的以太网线已经无线关闭,这打破了OnMessage与Service Bus队列的连接。等待10分钟后,我重新插入以太网电缆,OnMessage立即开始处理排队的元素。
  • On Message令人惊讶地相当稳定。它已经在global.asax.cs App Start中运行,缩短了几天,Factory IdleTimeout设置为24小时,而没有重新启动Web应用程序72小时。

总而言之,我现在要继续使用OnMessage / OnMessageAsync并密切关注它。如果我发现有些问题会改变我对OnMessage的看法,我会更新这个。

除此之外 - 确保您是否在Azure网站中使用OnMessage进行永久收听,并将其设置为&#34; Always On&#34;配置选项为&#34; On&#34;。否则,除非Web请求进入,否则OnMessage将被处理,并且在HTTP请求重新唤醒Web应用程序之前将不再处理消息。