Я пытаюсь добавить настраиваемую функцию безопасности, при которой клиент добавляет токен к каждой службе вызов, сделанный клиентом Silverlight, а затем служба может получить доступ к этому токену для управления системой безопасности приложения. Я пытаюсь сделать это, реализуя интерфейс IClientMessageInspector и связывая его с моим созданным клиентом службы. Я использую не прокси, созданный Visual Studio, а свой собственный клиент, созданный ChannelFactory. Мне удалось найти несколько решений в Интернете, которые, похоже, используют один из двух основных методов. Во-первых, добавив заголовок в коллекцию заголовков сообщения, представленного в IClientMessageInspector.BeforeSendRequest, а во-вторых, используя HttpRequestMessageProperty, чтобы добавить информацию в качестве свойства сообщения, представленного в IClientMessageInspector.BeforeSendRequest. Я безуспешно пробовал оба этих метода. Похоже, что оба метода успешно добавляют данные в запрос, но мне не удалось получить доступ ни к одному из них на сервере. Я бы добавил, что это совершенно новая для меня область, и вполне возможно, что я пропустил ответ в Интернете из-за неопытности.
Код для создания моего клиента:
private ISecurityAdministrationContract CreateChannel()
{
if (factory == null)
{
lock (this)
{
// Create a custom binding that uses HTTP and binary encoding.
var elements = new List<BindingElement>();
elements.Add(new BinaryMessageEncodingBindingElement());
elements.Add(new HttpTransportBindingElement());
var binding = new CustomBinding(elements);
// Create a channel factory for the service endpoint configured with the custom binding.
factory = new ChannelFactory<ISecurityAdministrationContract>(binding, new EndpointAddress(SecurityAdminServiceAddress));
//Add my IClientMessageInspector
factory.Endpoint.Behaviors.Add(new ClientSecurityInterceptor());
}
}
ISecurityAdministrationContract client = factory.CreateChannel();
return client;
}
}
ISecurityAdministrationContract client = factory.CreateChannel();
return client;
}
Код для добавления моего заголовка к запросу:
public object BeforeSendRequest(ref System.ServiceModel.Channels.Message request, System.ServiceModel.IClientChannel channel)
{
//Method 1
MessageHeader header = MessageHeader.CreateHeader("MyFirstAuthentication", "ns", "AAAAAAAA-BBBB-CCCC-DDDD-EEEEEEEEEEEEE");
request.Headers.Add(header);
//Method 2
HttpRequestMessageProperty httpRequestMessage;
httpRequestMessage = new HttpRequestMessageProperty();
httpRequestMessage.Headers["MySecondAuthentication"] = "11111111-2222-3333-4444-5555555555555";
request.Properties.Add(HttpRequestMessageProperty.Name, httpRequestMessage);
return null;
}
Приведенный выше код реализует оба метода. Принять: / Referer: ht_tp: //ipv4.fiddler: 6785 / ClientBin / Civica.Housing.xap Accept-Language: en-gb Длина содержимого: 400 Тип содержимого: приложение / мыло + msbin1 MySecondAuthentication: 11111111-2222-3333-4444-5555555555555 Принятие кодировки: gzip, deflate Пользовательский агент: Mozilla / 4.0 (совместимый; MSIE 8.0; Windows NT 5.1; Trident / 4.0; GTB5; InfoPath.2; .NET CLR 3.0.04506.648; .NET CLR 3.5.21022; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729; .NET4.0C; .NET4.0E; .NET CLR 2.0.50727) Хост: 127.0.0.1:6785 Подключение: Keep-Alive Pragma: без кеширования
V_ s _a_V_D Ahttp: //tempuri.org/ISecurityAdministrationContract/CreateAdvocateD ߔ_ 9 HJ 9 - D, D * @ MyFirstAuthentication _ns % AAAAAAAA-BBBB-CCCC-DDDD-EEEEEEEEEEEEED _ @ http: //ipv4.fiddler: 6785 / SecurityAdministrationRelayService.svc / V @_CreateAdvocate_ http://tempuri.org/@ AdvocateVO {"Id": 0, "UserId": 4, "AdvocateForId": 8, "ValidFrom": "/ Date (1291127869690 + 0000) /", "ValidTo": null} __
Это похоже, содержит информацию о токене для обоих методов.
Проблема возникает на сервере, когда я пытаюсь извлечь эту информацию. Я реализовал IOperationInvoker и безуспешно пытался найти заголовок в IncomingMessageHeaders. Я также искал в IncomingMessageProperties, но я не вижу никаких добавленных мной деталей заголовка. Это код, который я использую в IOperationInvoker:
public object Invoke(object instance, object[] inputs, out object[] outputs)
{
PreInvoke(instance, inputs);
object returnedValue = null;
object[] outputparams = new object[] { };
Exception exception = null;
try
{
returnedValue = originalInvoker.Invoke(instance, inputs, out outputparams);
outputs = outputparams;
return returnedValue;
}
catch (Exception e)
{
exception = e; throw;
}
finally
{
PostInvoke(instance, returnedValue, outputparams, exception);
}
}
protected virtual void PreInvoke(object instance, object[] inputs)
{
//Look for header directly
int index = System.ServiceModel.OperationContext.Current.IncomingMessageHeaders.FindHeader("MyFirstAuthentication", "ns");
//Search via enumerator
foreach (var header in System.ServiceModel.OperationContext.Current.IncomingMessageHeaders)
{
}
}
FindHeader возвращает -1, в то время как перечислитель находит 5 заголовков, а именно; 'Action', 'MessageID', 'ReplyTo', 'VsDebuggerCausalityData' и 'To'.
Коллекция OperationContext.Current.IncomingMessageProperties содержит 4 записи, а именно: 'Via', 'Security', 'Encoder' и 'System .ServiceModel.Channels.RemoteEndpointMessageProperty '
Фактически, если я закомментирую строку в моем клиенте, которая добавляет IClientMessageInspector в ClienChannel, то http, который скрипач сообщает об изменениях, опуская добавленные детали, но заголовок и коллекции свойств на входящем сообщение не изменилось.
Любые идеи относительно того, как я могу получить доступ к этой информации, или почему она не представлена как часть IncomingMessage, будут с благодарностью приняты.