Supporting WSE plain text password with WCF BasicHttpBinding

November 16, 2011

Hi all,

Ok, so I did a bit of googling to see if this had been done by someone else, turns out I failed to find a suitable solution, just many frustrated people.

So this post is an attempt to make those people a little happier.

The solutions is as follows(, It’s a bit rough around the edges at the moment as I have just got it working and have not yet cleaned up the code).

  • Firstly, I created a binding to manage the header

    public class MyBehavior : BehaviorExtensionElement, IEndpointBehavior

    {      
    
        public MyBehavior(string userName, string password)
    
        {
    
            this.UserName = userName;
    
            this.Password = password;
    
        }
    
        #region IEndpointBehavior Members
    
        public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
    
        {
    
        }
    
        public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
    
        {
    
            clientRuntime.MessageInspectors.Add(new MyMessageInspector(this.UserName, this.Password));
    
        }
    
        public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher) 
    
        {
    
        }
    
        public void Validate(ServiceEndpoint endpoint) 
    
        { 
    
        }
    
        #endregion
    
        public override Type BehaviorType
    
        {
    
            get 
    
            { 
    
                return typeof(MyBehavior); 
    
            } 
    
        }
    
        protected override object CreateBehavior() 
    
        { 
    
            return new MyBehavior(this.UserName, this.Password); 
    
        }
    
        public string UserName { get; set; }
    
        public string Password { get; set; }
    
    }

Ok so now we can see this behavior adds a MessageInspector to every message. lets take a look at what the message inspector does.

  • MessageInspector

    class MyMessageInspector : IClientMessageInspector { public MyMessageInspector(string username, string password) { _username = username; _password = password; }

    public void AfterReceiveReply(ref System.ServiceModel.Channels.Message reply, object correlationState)
    {
            
    }
    
    public object BeforeSendRequest(ref System.ServiceModel.Channels.Message request, System.ServiceModel.IClientChannel channel)
    {
        var header = new WseHeader(_username, _password);            
                
        request.Headers.Add(header); 
            
        return null;
    }
    
    private string _username;
    private string _password;

    }

So here in my message inspector I add a new header.

In fact it’s this header that was making life hard for most people.

  • WseHeader

    class WseHeader : MessageHeader

    {        
    
        public WseHeader(string userName, string password)
    
        {
    
            this.UserName = userName;
    
            this.Password = password;
    
        }
    
        public string UserName
    
        {
    
            get;
    
            private set;
    
        }
    
        private string Password
    
        {
    
            get;
    
            set;
    
        }
    
        protected override void OnWriteStartHeader(System.Xml.XmlDictionaryWriter writer, MessageVersion messageVersion)
    
        {
    
            base.OnWriteStartHeader(writer, messageVersion);
    
            writer.WriteAttributeString("s:mustUnderstand", "0");
    
            writer.WriteAttributeString("xmlns:wsse", WsseNamespaceToken);
    
            writer.WriteAttributeString("xmlns:s", "http://schemas.xmlsoap.org/soap/envelope/");
    
        }
    
        protected override void OnWriteHeaderContents(System.Xml.XmlDictionaryWriter writer, MessageVersion messageVersion)
    
        {
    
            writer.WriteRaw(Properties.Resources.WseHeaderText.Replace("{USERNAME}", 
    
                this.UserName).Replace("{PASSWORD}", this.Password));
    
        }
    
        public override string Name
    
        {
    
            get { return "wsse:Security"; }
    
        }
    
        public override string Namespace
    
        {
    
            get { return ""; }
    
        }
    
        public override bool MustUnderstand
    
        {
    
            get
    
            {
    
                return false;
    
            }
    
        }
    
        private const string WsseNamespaceToken = "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd";
    
    }

This class will create a header like this

<wsse:Security s:mustUnderstand="0" xmlns:wsse="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd" xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">

    <wsse:UsernameToken wsu:Id="SecurityToken-3f7f983f-66ce-480d-bce6-170632d33f92" xmlns:wsu="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd">

        <wsse:Username>bek@anchorwsse:Username>

        <wsse:Password Type="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText">dotnetrockswsse:Password>

    wsse:UsernameToken>

    wsse:Security>

I just replace the username and password in code in the MessageHeader. I could probably do all this neater with the API but it’s good enough for my investigation tonight, I usually just add the WSE header directly into my configuration file and not bother with the behavior.

e.g.

<client>

          <endpoint address="http://anchor:8083/gdm/TemplateActionsService/TemplateActionsService" binding="basicHttpBinding" bindingConfiguration="TemplateActionsServiceSoapBinding" contract="TemplateActionsProxy.TemplateActionsServiceType" name="TemplateActionsServicePort">

          endpoint>

        client>

So I hope this helps somebody else. Ninja