首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Nservice bus sagas实现

Nservice bus sagas实现
EN

Stack Overflow用户
提问于 2013-08-01 00:24:03
回答 1查看 2.2K关注 0票数 0

我有一个nservice总线项目,我将其称为连接器。我的连接器接收各种类型的消息,例如ClientChangeMessage、ClientContactChangeMessage。我的连接器没有实现saga,所以我对每条消息都有处理程序,所以对于ClientChangeMessage,我有ClientChangeMessageHandler,当connector接收到ClientChangeMessage时会触发它,当我接收到ClientContactChangeMessage时会触发ClientContactChangeMessageHandler。

现在,在查看sagas实现时,我发现自己编写了以下代码(如果客户联系消息出现在ClientChange消息之前,即数据库中不存在客户端):

代码语言:javascript
复制
public class ClientContactChangeMessageHandler : ClientMessageHandler, 
    IHandleMessages<ClientContactChangeMessage>, 
    IAmStartedByMessages<ClientContactChangeMessage>, 
    IHandleMessages<ClientChangeMessage>
{
    [SetterProperty]
    public IClientContactChangeDb ClientContactChangeDb{get;set;}

    [SetterProperty]
    public IBusRefTranslator BusRefTranslator{get;set;}

    static ClientContactChangeMessageHandler()
    {
        Logger = LogManager.GetLogger(typeof (ClientContactChangeMessageHandler));
    }

    static ILog Logger;

    public void Handle(ClientContactChangeMessage message)
    {
        //Some handling logic
    }

    public void Handle(ClientChangeMessage message)
    {
        throw new NotImplementedException();
    }

    public override void ConfigureHowToFindSaga()
    {
        ConfigureMapping<ClientContactChangeMessage>(s => s.Id, m => m.Id);
        ConfigureMapping<ClientChangeMessage>(s => s.Id, m => m.Id);
        // Notice that we have no mappings for the OrderAuthorizationResponseMessage message. This is not needed since the HR
        // endpoint will do a Bus.Reply and NServiceBus will then automatically correlate the reply back to
        // the originating saga
    }
}


public class ClientMessageHandler : BaseMessageHandler
{
}

public class BaseMessageHandler : Saga<MySagaData>
{
}

public class MySagaData : IContainSagaData
{
    public Guid Id { get; set; }
    public string Originator { get; set; }
    public string OriginalMessageId { get; set; }
}

从示例中可以看出,我现在还必须实现CLientChangeMessage的句柄,现在我已经为我的ClientChangeMessage定义了一个处理程序,我是否必须在这里再次处理它,因为如果ClientChangeMessage在时间上确实到来了,我希望它被ClientChangeMessageHandler nad捕获并处理,而不是这个。

如果且仅当我在数据库中找不到客户端的本地引用时,我才会存储消息。看看网络上的saga的例子,我没有看到任何特定的地方或条件,这将被处理。我希望我将消息存储在ClientContactChange句柄方法中。

任何帮助都将不胜感激,

谢谢

更新:

似乎我没有正确地理解如何实现NService Bus Saga。据我所说,我在这里犯的错误是,我认为客户联系方式变更是一个单一的实体,即独立于客户变更消息。因此,我认为仅仅为了改变客户联系方式而实现Saga是错误的。以下是我必须更改代码的方式:

代码语言:javascript
复制
 public class ClientSaga : Saga<ClientSagaState>, 
        IAmStartedByMessages<ClientChangeMessage>, 
        IAmStartedByMessages<ClientContactChangeMessage>, 
        IHandleTimeout<ClientSagaState>
    {
        [SetterProperty]
        public IClientContactChangeDb ClientContactChangeDb{get;set;}

        [SetterProperty]
        public IBusRefTranslator BusRefTranslator{get;set;}



        public void Handle(ClientContactChangeMessage message)
        {
            //Some handling logic
            //Check if client is not in database then store the state
            this.ClientContactChange=message;
            //if client is in the data base then 
            MarkAsComplete();
        }

        public void Handle(ClientChangeMessage message)
        {
            //Update or create the client depending on the situation
            //check for dependencies
            if(this.ClientContactChange !=null)
            {
                 //Handle the contact change
            }
        }

        public override void ConfigureHowToFindSaga()
        {
            ConfigureMapping<ClientContactChangeMessage>(s => s.ClientRef, m => m.ClientRef);
            ConfigureMapping<ClientChangeMessage>(s => s.ClienttnRef, m => m.Id);
            // Notice that we have no mappings for the OrderAuthorizationResponseMessage message. This is not needed since the HR
            // endpoint will do a Bus.Reply and NServiceBus will then automatically correlate the reply back to
            // the originating saga
        }
    }


    public class ClientSagaState: IContainSagaData
    {
        //i dont need these three fields
        public Guid Id { get; set; }
        public string Originator { get; set; }
        public string OriginalMessageId { get; set; }

       // the fields which i needed
       public Guid ClientRef {gee; set;}
       public ClientChangeMessage ClientChange {get;set;}
       public ClientContactChange ClientContactChange {get;set;}

    }
EN

回答 1

Stack Overflow用户

发布于 2013-08-02 03:22:07

由于两个处理程序处理相同的消息类型,因此将调用这两个处理程序。如果您愿意,可以使用ISpecifyMessageHandlerOrdering指定调用它们的顺序。此外,您可以根据可能解决次要问题的条件来短路此链。

如果这不起作用,您可能需要考虑对消息进行版本控制,以便以一种优雅的方式支持这两种场景。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/17975808

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档