2 // System.ServiceModel.MessageHeader.cs
4 // Author: Duncan Mak (duncan@novell.com)
5 // Atsushi Enomoto (atsushi@ximian.com)
7 // Copyright (C) 2005 Novell, Inc (http://www.novell.com)
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 using System.Runtime.Serialization;
31 using System.ServiceModel;
32 using System.ServiceModel.Channels;
36 namespace System.ServiceModel.Channels
38 public abstract class MessageHeader : MessageHeaderInfo
40 static readonly XmlWriterSettings writer_settings;
42 static MessageHeader ()
44 writer_settings = new XmlWriterSettings ();
45 writer_settings.OmitXmlDeclaration = true;
46 writer_settings.Indent = true;
49 protected MessageHeader () {}
51 static string default_actor = String.Empty;
52 static bool default_is_ref = false;
53 static bool default_must_understand = false;
54 static bool default_relay = false;
55 static Type [] knownTypes = new Type [1] {typeof (EndpointAddress10)};
57 internal static MessageHeader CreateInternalHeader (XmlElement el, string soap_ns)
59 return new RawMessageHeader (el, soap_ns);
62 public static MessageHeader CreateHeader (string name, string ns, object value)
64 return CreateHeader (name, ns, value, default_must_understand);
67 public static MessageHeader CreateHeader (string name, string ns, object value, bool must_understand)
69 return CreateHeader (name, ns, value, must_understand, default_actor);
72 public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter)
74 return CreateHeader (name, ns, value, formatter, default_must_understand,
75 default_actor, default_relay);
78 public static MessageHeader CreateHeader (string name, string ns, object value,
79 bool must_understand, string actor)
81 return CreateHeader (name, ns, value, must_understand, actor, default_relay);
84 public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter,
87 return CreateHeader (name, ns, value, formatter, must_understand, default_actor, default_relay);
90 public static MessageHeader CreateHeader (string name, string ns, object value,
91 bool must_understand, string actor, bool relay)
93 return CreateHeader (name, ns, value, new DataContractSerializer (value.GetType (), knownTypes),
94 must_understand, actor, relay);
97 public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter,
98 bool must_understand, string actor)
100 return CreateHeader (name, ns, value, formatter, must_understand, actor, default_relay);
103 public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter,
104 bool must_understand, string actor, bool relay)
106 // FIXME: how to get IsReferenceParameter ?
107 return new DefaultMessageHeader (name, ns, value, formatter, default_is_ref, must_understand, actor, relay);
110 public virtual bool IsMessageVersionSupported (MessageVersion version)
112 if (version.Envelope == EnvelopeVersion.Soap12)
113 if (Actor == EnvelopeVersion.Soap11.NextDestinationActorValue)
116 if (version.Envelope == EnvelopeVersion.Soap11)
117 if (Actor == EnvelopeVersion.Soap12.NextDestinationActorValue ||
118 Actor == EnvelopeVersion.Soap12UltimateReceiver)
121 // by default, it's always supported
125 protected abstract void OnWriteHeaderContents (XmlDictionaryWriter writer, MessageVersion version);
127 protected virtual void OnWriteStartHeader (XmlDictionaryWriter writer, MessageVersion version)
129 writer.WriteStartElement (this.Name, this.Namespace);
130 WriteHeaderAttributes (writer, version);
133 public override string ToString ()
135 StringBuilder sb = new StringBuilder ();
137 XmlWriter w = XmlWriter.Create (sb, writer_settings);
139 WriteHeader (w, MessageVersion.Default);
142 return sb.ToString ();
145 public void WriteHeader (XmlDictionaryWriter writer, MessageVersion version)
148 throw new ArgumentNullException ("writer is null.");
151 throw new ArgumentNullException ("version is null.");
153 if (version.Envelope == EnvelopeVersion.None)
156 WriteStartHeader (writer, version);
157 WriteHeaderContents (writer, version);
159 writer.WriteEndElement ();
162 public void WriteHeader (XmlWriter writer, MessageVersion version)
164 WriteHeader (XmlDictionaryWriter.CreateDictionaryWriter (writer), version);
167 protected void WriteHeaderAttributes (XmlDictionaryWriter writer, MessageVersion version)
170 writer.WriteAttributeString ("u", "Id", Constants.WsuNamespace, Id);
171 if (Actor != String.Empty) {
172 if (version.Envelope == EnvelopeVersion.Soap11)
173 writer.WriteAttributeString ("s", "actor", version.Envelope.Namespace, Actor);
175 if (version.Envelope == EnvelopeVersion.Soap12)
176 writer.WriteAttributeString ("s", "role", version.Envelope.Namespace, Actor);
179 // mustUnderstand is the same across SOAP 1.1 and 1.2
180 if (MustUnderstand == true)
181 writer.WriteAttributeString ("s", "mustUnderstand", version.Envelope.Namespace, "1");
183 // relay is only available on SOAP 1.2
184 if (Relay == true && version.Envelope == EnvelopeVersion.Soap12)
185 writer.WriteAttributeString ("s", "relay", version.Envelope.Namespace, "true");
188 public void WriteHeaderContents (XmlDictionaryWriter writer, MessageVersion version)
190 this.OnWriteHeaderContents (writer, version);
193 public void WriteStartHeader (XmlDictionaryWriter writer, MessageVersion version)
195 this.OnWriteStartHeader (writer, version);
198 public override string Actor { get { return default_actor; }}
200 public override bool IsReferenceParameter { get { return default_is_ref; }}
202 public override bool MustUnderstand { get { return default_must_understand; }}
204 public override bool Relay { get { return default_relay; }}
206 internal class RawMessageHeader : MessageHeader
210 bool is_ref, must_understand, relay;
213 public RawMessageHeader (XmlElement source, string soap_ns)
215 this.source = source;
217 Id = source.HasAttribute ("Id", Constants.WsuNamespace) ?
218 source.GetAttribute ("Id", Constants.WsuNamespace) :
221 // FIXME: fill is_ref
222 string s = source.GetAttribute ("relay", soap_ns);
223 relay = s.Length > 0 ? XmlConvert.ToBoolean (s) : false;
224 s = source.GetAttribute ("mustUnderstand", soap_ns);
225 must_understand = s.Length > 0 ? XmlConvert.ToBoolean (s) : false;
226 actor = source.GetAttribute ("actor", soap_ns);
229 public XmlReader CreateReader ()
231 return new XmlNodeReader (source);
234 protected override void OnWriteHeaderContents (
235 XmlDictionaryWriter writer, MessageVersion version)
237 source.WriteContentTo (writer);
240 public override string Actor { get { return actor; }}
242 public override bool IsReferenceParameter { get { return is_ref; }}
244 public override bool MustUnderstand { get { return must_understand; }}
246 public override string Name { get { return source.LocalName; }}
248 public override string Namespace { get { return source.NamespaceURI; }}
250 public override bool Relay { get { return relay; }}
253 internal class DefaultMessageHeader : MessageHeader
255 string actor, name, ns;
257 XmlObjectSerializer formatter;
258 bool is_ref, must_understand, relay;
260 internal DefaultMessageHeader (string name, string ns, object value, XmlObjectSerializer formatter,
261 bool isReferenceParameter,
262 bool mustUnderstand, string actor, bool relay)
267 this.formatter = formatter;
268 this.is_ref = isReferenceParameter;
269 this.must_understand = mustUnderstand;
274 protected override void OnWriteHeaderContents (XmlDictionaryWriter writer,
275 MessageVersion version)
277 if (value is EndpointAddress)
278 ((EndpointAddress) value).WriteTo (version.Addressing, writer, name, ns);
280 this.formatter.WriteObjectContent (writer, value);
283 public override string Actor { get { return actor; }}
285 public override bool IsReferenceParameter { get { return is_ref; }}
287 public override bool MustUnderstand { get { return must_understand; }}
289 public override string Name { get { return name; }}
291 public override string Namespace { get { return ns; }}
293 public override bool Relay { get { return relay; }}