操纵WCF标头详细信息

时间:2012-04-17 10:50:41

标签: c# wcf security header

我正在为某些STS服务构建一个客户端,现在我试图在WCF消息中添加一个Header超过一天。在我对RequestSecurityToken的调用中,我必须包含一个UsernameToken。

我不确定如何做到这一点。目前我定义了一个端点行为和一个消息检查器(花了我足够长的时间来发现那些......)。在后者的BeforeSendRequest()中,我创建了一个自定义类“Security”的对象,该对象派生自MessageHeader。安全性包括UsernameToken的实例。

public class MessageInspector : IClientMessageInspector {

 public object BeforeSendRequest(ref Message request, IClientChannel channel) {
    Security uns = new Security();
    uns.UsernameToken = new UsernameToken();

    // ...

    var Header = new MessageHeader<Security>(uns);
    var untyped = Header.GetUntypedHeader("Security", "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd");
    request.Headers.Add(untyped);
    return null;
 }
}

public class Security : MessageHeader {
 public UsernameToken UsernameToken = new UsernameToken();

 public override string Name {
    get { return "Security"; }
 }

 public override string Namespace {
    get { return "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd"; }
 }
}

public class UsernameToken {
 public String Username = "";
 public Password Password = new Password();
}

这就是被序列化的内容

<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
  <s:Header>
    <Action s:mustUnderstand="1" xmlns="http://schemas.microsoft.com/ws/2005/05/addressing/none">urn:RequestSecurityToken</Action>
    <Security xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd">
      <UsernameToken xmlns="http://schemas.datacontract.org/2004/07/Tarifrechner.Kfz">
        <Password>
          <Type>http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText</Type>
          <password>******</password>
        </Password>
        <Username>******</Username>
      </UsernameToken>
    </Security>
  </s:Header>
  <s:Body />
</s:Envelope>

特别是UsernameToken的命名空间似乎是错误的。我知道它来自数据合同序列化,但我需要一个不同的命名空间。

这就是我希望序列化数据看起来像

<?xml version="1.0" encoding="utf-8"?>
<soap:Envelope xmlns:soap="...">
  <soap:Header>
    <Security xmlns:q1="http://www.bipro.net/namespace" xsi:type="q1:UserNameSecurity" 
          xmlns="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd">
      <UsernameToken>
        <Username>******</Username>
        <Password Type="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText">******</Password>
      </UsernameToken>
    </Security>
    <wsa:Action>urn:RequestSecurityToken</wsa:Action>
    <wsse:Security>
      <wsu:Timestamp wsu:Id="Timestamp-b9dd599d-5901-451d-8321-6a309091f273">
        <wsu:Created>2012-03-11T16:02:56Z</wsu:Created>
        <wsu:Expires>2012-03-11T16:07:56Z</wsu:Expires>
      </wsu:Timestamp>
    </wsse:Security>
  </soap:Header>
  <soap:Body>
    <RequestSecurityToken xmlns="http://schemas.xmlsoap.org/ws/2005/02/trust">
      <TokenType>http://schemas.xmlsoap.org/ws/2005/02/sc/sct</TokenType>
      <RequestType>
        http://schemas.xmlsoap.org/ws/2005/02/trust/Issue
      </RequestType>
    </RequestSecurityToken>
  </soap:Body>
</soap:Envelope>

我的做法是对的吗?我如何操作标题详细信息的命名空间或数据是否被序列化为属性或元素?

更新


正如Ladislav已经指出的那样,我不必自己实现UsernameToken这样的类。我这样做只是因为我对WCF的了解非常有限。

到目前为止,我发现,WS2007HttpBinding配置为使用SecurityMode.TransportWithMessageCredential并且EstablishSecurityContext设置为false,几乎生成我正在寻找的XML。我该怎么知道呢?

但是还有一个问题:我的请求有一个空的body元素,我想要生成的请求,在body元素中有一个RequestSecurityToken元素。有谁知道,我怎么能做到这一点?

使用EstablishSecurityContext = true有帮助,但同时将我的Soap-Action从所需的“urn:RequestSecurityToken”更改为非工作的“http://docs.oasis-open.org/ws-sx/ws-信任/ 200512 / RST / SCT”。


我感谢任何回答!

非常感谢!

的Björn

2 个答案:

答案 0 :(得分:1)

另一种方法是为您的请求定义 MessageContract 类型,这允许您定义SOAP消息的标头和正文中显示的内容并调整使用的命名空间。例如,请考虑以下服务定义:

[ServiceContract]
    public interface IMyService
    {
        [OperationContract]
        MyResponse DoSomething(MyRequest request);
    }

    public class MyService : IMyService
    {
        public MyResponse DoSomething(MyRequest request)
        {
            return new MyResponse()
            {
                Details = "Service did something awesome.",
                Timestamp = DateTime.Now
            };
        }
    }

    [MessageContract(IsWrapped = true, WrapperNamespace = "http://myservice/messages/")]
    public class MyRequest
    {
        [MessageHeader(Namespace = "http://myservice/security")]
        public string TokenThingy
        {
            get;
            set;
        }
    }

    [MessageContract(IsWrapped = true, WrapperNamespace = "http://myservice/messages")]
    public class MyResponse
    {
        [MessageBodyMember]
        public string Details
        {
            get;
            set;
        }

        [MessageBodyMember]
        public DateTime Timestamp
        {
            get;
            set;
        }
    }

发送请求会产生以下SOAP:

<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
  <s:Header>
    <Action s:mustUnderstand="1" xmlns="http://schemas.microsoft.com/ws/2005/05/addressing/none">http://tempuri.org/IMyService/DoSomething</Action>
    <h:TokenThingy xmlns:h="http://myservice/security">fda</h:TokenThingy>
  </s:Header>
  <s:Body>
    <MyRequest xmlns="http://myservice/messages/" />
  </s:Body>
</s:Envelope>

该服务的回复如下:

<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
  <s:Header />
  <s:Body>
    <MyResponse xmlns="http://myservice/messages">
      <Details xmlns="http://tempuri.org/">Service did something awesome.</Details>
      <Timestamp xmlns="http://tempuri.org/">2012-05-04T17:04:36.5980424-04:00</Timestamp>
    </MyResponse>
  </s:Body>
</s:Envelope>

答案 1 :(得分:0)

根据您需要如何控制标题内容以及需要插入标题的位置,有几种不同的方法可以向邮件添加标题。

在您的应用程序代码中,您可以围绕请求创建一个OperationContextScope,以便更改某些请求属性。在OperationContextScope中,您有一个有效的OperationContext.Current实例,它允许通过OutgoingMessageHeaders集合操作消息头。使用此方法可深入控制标头到应用程序代码中。您必须负责在需要的地方复制适当的代码。

These two链接(来自WCF团队的某个人)通过大量代码示例更详细地讨论了这一点: