使用Azure Service Fabric的默认客户端时如何在请求中添加消息头? [英] How to add message header to the request when using default client of Azure service fabric?
问题描述
我想知道是否可以在传出的请求中注入自定义消息头,以携带附加信息,而无需反序列化有效负载,以完全实现诸如messagesinspector提供的诸如wcf之类的请求的身份验证,验证或相关性?
I am wondering it is possible to inject custom message header to outgoing request to carry additional information without deserialize the payload to fullfill the functionality like authentication, validation or correlation of request like wcf provided by means of messagesinspector?
推荐答案
更新
有了SDK v2,您现在可以(相对)轻松地修改可靠服务和参与者的标头。请注意,在下面的示例中,为简洁起见,省略了一些包装器成员。
With SDK v2 you can now (relatively) easily modify the headers of both Reliable Services and Actors. Note in the examples below some wrapper members were omitted for brevity.
客户
我们使用 ServiceProxyFactory
创建代理,而不是静态的 ServiceProxy
。然后,我们可以包装 IServiceRemotingClientFactory
和 IServiceRemotingClient
并截获服务调用。使用 ActorProxyFactory
可以完成相同的操作。请注意,这会覆盖属性,例如 WcfServiceRemotingProviderAttribute
的行为,因为我们自己明确指定了客户端工厂。
We use ServiceProxyFactory
to create proxies instead of the static ServiceProxy
. Then we can wrap IServiceRemotingClientFactory
and IServiceRemotingClient
and intercept the service calls. The same can be done with ActorProxyFactory
. Note this overrides the behavior of attributes such as the WcfServiceRemotingProviderAttribute
, since we explicitly specify the client factory ourselves.
_proxyFactory = new ServiceProxyFactory(c => new ServiceRemotingClientFactoryWrapper(
// we can use any factory here
new WcfServiceRemotingClientFactory(callbackClient: c)));
private class ServiceRemotingClientFactoryWrapper : IServiceRemotingClientFactory
{
private readonly IServiceRemotingClientFactory _inner;
public ServiceRemotingClientFactoryWrapper(IServiceRemotingClientFactory inner)
{
_inner = inner;
}
public async Task<IServiceRemotingClient> GetClientAsync(Uri serviceUri, ServicePartitionKey partitionKey, TargetReplicaSelector targetReplicaSelector,
string listenerName, OperationRetrySettings retrySettings, CancellationToken cancellationToken)
{
var client = await _inner.GetClientAsync(serviceUri, partitionKey, targetReplicaSelector, listenerName, retrySettings, cancellationToken).ConfigureAwait(false);
return new ServiceRemotingClientWrapper(client);
}
}
private class ServiceRemotingClientWrapper : IServiceRemotingClient
{
private readonly IServiceRemotingClient _inner;
public ServiceRemotingClientWrapper(IServiceRemotingClient inner)
{
_inner = inner;
}
public Task<byte[]> RequestResponseAsync(ServiceRemotingMessageHeaders messageHeaders, byte[] requestBody)
{
// use messageHeaders.AddHeader() here
return _inner.RequestResponseAsync(messageHeaders, requestBody);
}
public void SendOneWay(ServiceRemotingMessageHeaders messageHeaders, byte[] requestBody)
{
// use messageHeaders.AddHeader() here
_inner.SendOneWay(messageHeaders, requestBody);
}
}
服务器
从 ServiceRemotingDispatcher
和 ActorServiceRemotingDispatcher
继承以检查标头。
class CustomServiceRemotingDispatcher : ServiceRemotingDispatcher
{
public override async Task<byte[]> RequestResponseAsync(IServiceRemotingRequestContext requestContext, ServiceRemotingMessageHeaders messageHeaders, byte[] requestBody)
{
// read messageHeaders here
// or alternatively put them in an AsyncLocal<T> scope
// so they can be accessed down the call chain
return base.RequestResponseAsync(requestContext, messageHeaders, requestBody);
}
}
要使用此类,我们同样需要重写通过直接创建通信侦听器来 ServiceRemotingProviderAttribute
:
To use this class, again we need to override the ServiceRemotingProviderAttribute
by directly creating the communication listener:
class MyService : StatelessService
{
protected override IEnumerable<ServiceInstanceListener> CreateServiceInstanceListeners()
{
yield return new ServiceInstanceListener(context => new WcfServiceRemotingListener(context, new CustomServiceRemotingDispatcher());
}
}
这篇关于使用Azure Service Fabric的默认客户端时如何在请求中添加消息头?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!