通过ConsumerContext--Masstransit发布命令处理程序的命令结果

本文关键字:命令 结果 命令处理程序 ConsumerContext--Masstransit 通过 | 更新日期: 2023-09-27 18:28:35

我已经为masstransit配置了一个webapi控制器。像这样的东西。

_busControl = MassTransit.Bus.Factory.CreateUsingRabbitMq(x =>
            {
                x.Host(new Uri("rabbitmq://localhost/"), h =>{ });
            });
            _busControl.Start();

控制台应用程序是我的服务器,它正在监听我的命令/事件--

 var BusControl = Bus.Factory.CreateUsingRabbitMq(cfg =>
             {
                 var host = cfg.Host(new Uri("rabbitmq://localhost/"), h =>
                {
                    //h.Username("guest");
                    //h.Password("guest");
                });
                 cfg.ReceiveEndpoint(host, "create_product_queue", config => config.Consumer<CreateInventoryProductItemHandler>());
                 cfg.ReceiveEndpoint(host, "ProductItemCreatedEvent", config => config.Handler<ProductItemCreatedEvent>(async context => await Console.Out.WriteLineAsync($"Product Item Id -----{context.Message.ProductItemId} "))); <---- this never gets executed . 
             });
            BusControl.Start();

控制器的动作类似于

[HttpPost("product")]
        public async Task<IActionResult> Post([FromBody] Product Product)
        {
            var endpoint = await Startup.Bus.GetSendEndpoint(new Uri("rabbitmq://localhost/create_product_queue"));
            await endpoint.Send<CreateInventoryProductItem>(new
            {
                ProductName = Product.ProductName,
                PartNumber = Product.PartNumber,
                ManufactureDate = Product.ManufacturedDate
            });
            return new HttpStatusCodeResult(StatusCodes.Status202Accepted);
        }

处理程序是

public async Task Consume(ConsumeContext<CreateInventoryProductItem> context)
        {
            CreateInventoryProductItem Command = context.Message;
            Product Product = await Context.SaveAsync(new Product(Command.ProductName
                , Command.PartNumber
                , Command.ManufacturedDate));
            await context.Publish<ProductItemCreatedEvent>(Product.Id); <--- problem area
        }

现在我的场景是。

  1. 创建命令对象。

  2. 从新创建的实体框架Context.SaveAsync()方法中获取产品Id。

  3. 将ProductId发布到总线,以便其他侦听器可以侦听此产品Id并采取相应的操作。

    我正试图找出在这种情况下应用的最佳模式。

现在问题从这一行的-- await context.Publish<ProductItemCreatedEvent>(Product.Id);开始。它尝试一次又一次地无限地发布到同一队列。结果,同一个Product对象被插入了数百次(我知道这很蹩脚,因为没有唯一的约束,但这不是这里的问题),而发生的是,我的命令处理程序尽管是强类型的,但却被无限期地执行。如何摆脱这种情况。我对大众运输的概念还很陌生。

因此,有人可以揭示为什么会发生这种情况。

通过ConsumerContext--Masstransit发布命令处理程序的命令结果

所以问题是您需要将一个对象传递给您正在使用的Publish重载:

await context.Publish<ProductItemCreatedEvent>(Product.Id);

如果ProductItemCreatedEvent包含两个属性,IdDescription,您将创建一个匿名对象来初始化接口:

await context.Publish<ProductItemCreatedEvent>(new
{
    Id = Product.Id,
    Description = Product.Description,
});

通过这种方式,传递一个具有属性的对象,该对象可以为事件接口启动动态生成的支持类。