内部部署NServicebus应用程序从Azure Service Bus队列接收消息

时间:2012-10-10 09:48:03

标签: azure nservicebus servicebus azure-queues

我目前正努力在nServiceBus托管的应用程序上运行并运行。我有一个第三方发布消息的天蓝色ServiceBus队列,我希望我的应用程序(目前在本地托管)接收这些消息。

我已经搜索了有关如何配置端点的答案,但我在有效的配置中没有运气。有没有人这样做过,因为我可以找到如何连接Azure存储队列而不是servicebus队列的示例。 (出于其他原因,我需要azure servicebus队列)

我的配置如下

public void Init()
    {
        Configure.With()
           .DefaultBuilder()
           .XmlSerializer()
           .UnicastBus()
           .AzureServiceBusMessageQueue()
           .IsTransactional(true)
           .MessageForwardingInCaseOfFault()
           .UseInMemoryTimeoutPersister()
           .InMemorySubscriptionStorage();
    }

。     Message =启动端点时出现异常,已记录错误。原因:输入队列[mytimeoutmanager @ sb:// [ * ] .servicebus.windows.net /]必须与此Source = NServiceBus.Host

在同一台机器上

<configuration>
  <configSections>
    <section name="MessageForwardingInCaseOfFaultConfig" type="NServiceBus.Config.MessageForwardingInCaseOfFaultConfig, NServiceBus.Core" />
    <section name="UnicastBusConfig" type="NServiceBus.Config.UnicastBusConfig, NServiceBus.Core" />
    <section name="AzureServiceBusQueueConfig" type="NServiceBus.Config.AzureServiceBusQueueConfig, NServiceBus.Azure" />
    <section name="AzureTimeoutPersisterConfig" type="NServiceBus.Timeout.Hosting.Azure.AzureTimeoutPersisterConfig, NServiceBus.Timeout.Hosting.Azure" />
  </configSections>
  <AzureServiceBusQueueConfig IssuerName="owner" QueueName="testqueue" IssuerKey="[KEY]" ServiceNamespace="[NS]" />
  <MessageForwardingInCaseOfFaultConfig ErrorQueue="error" />
  <!-- Use the following line to explicitly set the Timeout manager address -->
  <UnicastBusConfig TimeoutManagerAddress="MyTimeoutManager" />
  <!-- Use the following line to explicity set the Timeout persisters connectionstring -->
  <AzureTimeoutPersisterConfig ConnectionString="UseDevelopmentStorage=true" />
  <startup useLegacyV2RuntimeActivationPolicy="true">
    <supportedruntime version="v4.0" />
    <requiredruntime version="v4.0.20506" />
    <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.0" />
  </startup>
</configuration>

2 个答案:

答案 0 :(得分:0)

尝试将UnicastBus()移至通话结束,如下所示:

    Configure.With()
       .DefaultBuilder()
       .XmlSerializer()
       .AzureServiceBusMessageQueue()
       .IsTransactional(true)
       .MessageForwardingInCaseOfFault()
       .UseInMemoryTimeoutPersister()
       .InMemorySubscriptionStorage()
       .UnicastBus(); // <- Here

关于那些向队列发布消息的第三方。请记住,他们需要尊重how NServiceBus handles serialization/deserialization。以下是在NServiceBus中完成的操作(最重要的部分是使用原始消息初始化BrokeredMessage,使用BinaryFormatter进行序列化的结果):

    private void Send(Byte[] rawMessage, QueueClient sender)
    {
        var numRetries = 0;
        var sent = false;

        while(!sent)
        {
            try
            {
                var brokeredMessage = new BrokeredMessage(rawMessage);

                sender.Send(brokeredMessage);

                sent = true;
            }
                // back off when we're being throttled
            catch (ServerBusyException)
            {
                numRetries++;

                if (numRetries >= MaxDeliveryCount) throw;

                Thread.Sleep(TimeSpan.FromSeconds(numRetries * DefaultBackoffTimeInSeconds));
            }
        }

    }

    private static byte[] SerializeMessage(TransportMessage message)
    {
        if (message.Headers == null)
            message.Headers = new Dictionary<string, string>();

        if (!message.Headers.ContainsKey(Idforcorrelation))
            message.Headers.Add(Idforcorrelation, null);

        if (String.IsNullOrEmpty(message.Headers[Idforcorrelation]))
            message.Headers[Idforcorrelation] = message.IdForCorrelation;

        using (var stream = new MemoryStream())
        {
            var formatter = new BinaryFormatter();
            formatter.Serialize(stream, message);
            return stream.ToArray();
        }
    }

如果您希望NServiceBus正确反序列化邮件,请确保您的各方正确序列化邮件。

答案 1 :(得分:0)

我现在遇到了完全相同的问题,并花了几个小时来弄清楚如何解决它。基本上,Azure超时持久性仅支持使用NServiceBus.Hosting.Azure的Azure托管端点。如果使用NServiceBus.Host进程来托管端点,则它使用NServiceBus.Timeout.Hosting.Windows命名空间类。它用MSMQ初始化了一个TransactionalTransport,你得到这条消息。

我使用了两种方法来避免它:

  1. 如果必须使用As_Server端点配置,则可以在初始化时使用.DisableTimeoutManager(),它将完全跳过TimeoutDispatcher初始化
  2. 使用As_Client端点配置,它不使用事务模式进行传输,并且超时调度程序未被初始化
  3. 可能有办法以某种方式注入Azure超时管理器,但我还没有找到它,我实际上需要As_Client,所以它对我来说很好。