我有一个在Masstransit中使用请求/响应消息的Web应用程序。 这适用于测试环境,没问题。
然而,在客户部署方面,我们遇到了问题。在客户站点,我们有两个网段A和B.执行数据库调用的组件位于段A,Web应用程序和段B中的RabbitMq服务器。
由于安全限制,A段中的组件必须通过具有给定地址的负载均衡器。组件本身可以通过Masstransit连接到RabbitMQ。到目前为止一切都很好。
B段上的Web组件使用RabbitMq服务器的直接地址。当web组件现在开始请求/响应调用时,我可以看到消息到达段A中的组件。 但是我看到消费者试图在"错误的"上调用RabbitMQ服务器。地址。它使用Web组件用于发出请求的地址。但是,A段中的组件应该回复" loadbalancer"地址。
有没有办法配置或告诉RespondAsync调用使用为该组件配置的连接地址?
当然最简单的方法是让Web组件也通过负载均衡器连接,但由于网段/安全设置,负载均衡器只能从A段到达。
感谢任何输入/帮助。
答案 0 :(得分:0)
我和rabbitmq联盟有类似的问题。这就是我所做的。
class ResponseAddressSendObserver : ISendObserver
{
private readonly string _hostUriString;
public ResponseAddressSendObserver(string hostUriString)
{
_hostUriString = hostUriString;
}
public Task PreSend<T>(SendContext<T> context)
where T : class
{
if (context.ResponseAddress != null)
{
// Send relative response address alongside the message
context.Headers.Set("RelativeResponseAddress",
context.ResponseAddress.AbsoluteUri.Substring(_hostUriString.Length));
}
return Task.CompletedTask;
}
...
}
class ResponseAddressConsumeFilter : IFilter<ConsumeContext>
{
private readonly string _hostUriString;
public ResponseAddressConsumeFilter(string hostUriString)
{
_hostUriString = hostUriString;
}
public Task Send(ConsumeContext context, IPipe<ConsumeContext> next)
{
var responseAddressOverride = GetResponseAddress(_hostUriString, context);
return next.Send(new ResponseAddressConsumeContext(responseAddressOverride, context));
}
public void Probe(ProbeContext context){}
private static Uri GetResponseAddress(string host, ConsumeContext context)
{
if (context.ResponseAddress == null)
return context.ResponseAddress;
object relativeResponseAddress;
if (!context.Headers.TryGetHeader("RelativeResponseAddress", out relativeResponseAddress) || !(relativeResponseAddress is string))
throw new InvalidOperationException("Message has ResponseAddress but doen't have RelativeResponseAddress header");
return new Uri(host + relativeResponseAddress);
}
}
class ResponseAddressConsumeContext : BaseConsumeContext
{
private readonly ConsumeContext _context;
public ResponseAddressConsumeContext(Uri responseAddressOverride, ConsumeContext context)
: base(context.ReceiveContext)
{
_context = context;
ResponseAddress = responseAddressOverride;
}
public override Uri ResponseAddress { get; }
public override bool TryGetMessage<T>(out ConsumeContext<T> consumeContext)
{
ConsumeContext<T> context;
if (_context.TryGetMessage(out context))
{
// the most hackish part in the whole arrangement
consumeContext = new MessageConsumeContext<T>(this, context.Message);
return true;
}
else
{
consumeContext = null;
return false;
}
}
// all other members just delegate to _context
}
配置总线时
var result = MassTransit.Bus.Factory.CreateUsingRabbitMq(cfg =>
{
var host = cfg.Host(new Uri(hostAddress), h =>
{
h.Username(...);
h.Password(...);
});
cfg.UseFilter(new ResponseAddressConsumeFilter(hostAddress));
...
});
result.ConnectSendObserver(new ResponseAddressSendObserver(hostAddress));
现在相对响应地址与消息一起发送并在接收方使用。
文档不建议使用观察者来修改任何内容,但在这种情况下应该没问题。
也许三个是更好的解决方案,但我还没找到。 HTH