MessageHeader.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. //
  2. // System.ServiceModel.MessageHeader.cs
  3. //
  4. // Author: Duncan Mak ([email protected])
  5. // Atsushi Enomoto ([email protected])
  6. //
  7. // Copyright (C) 2005 Novell, Inc (http://www.novell.com)
  8. //
  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:
  16. //
  17. // The above copyright notice and this permission notice shall be
  18. // included in all copies or substantial portions of the Software.
  19. //
  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.
  27. //
  28. using System;
  29. using System.IO;
  30. using System.Runtime.Serialization;
  31. using System.ServiceModel;
  32. using System.ServiceModel.Channels;
  33. using System.Text;
  34. using System.Xml;
  35. namespace System.ServiceModel.Channels
  36. {
  37. public abstract class MessageHeader : MessageHeaderInfo
  38. {
  39. static readonly XmlWriterSettings writer_settings;
  40. static MessageHeader ()
  41. {
  42. writer_settings = new XmlWriterSettings ();
  43. writer_settings.OmitXmlDeclaration = true;
  44. writer_settings.Indent = true;
  45. }
  46. protected MessageHeader () {}
  47. static string default_actor = String.Empty;
  48. static bool default_is_ref = false;
  49. static bool default_must_understand = false;
  50. static bool default_relay = false;
  51. public static MessageHeader CreateHeader (string name, string ns, object value)
  52. {
  53. return CreateHeader (name, ns, value, default_must_understand);
  54. }
  55. public static MessageHeader CreateHeader (string name, string ns, object value, bool must_understand)
  56. {
  57. return CreateHeader (name, ns, value, must_understand, default_actor);
  58. }
  59. public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter)
  60. {
  61. return CreateHeader (name, ns, value, formatter, default_must_understand,
  62. default_actor, default_relay);
  63. }
  64. public static MessageHeader CreateHeader (string name, string ns, object value,
  65. bool must_understand, string actor)
  66. {
  67. return CreateHeader (name, ns, value, must_understand, actor, default_relay);
  68. }
  69. public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter,
  70. bool must_understand)
  71. {
  72. return CreateHeader (name, ns, value, formatter, must_understand, default_actor, default_relay);
  73. }
  74. public static MessageHeader CreateHeader (string name, string ns, object value,
  75. bool must_understand, string actor, bool relay)
  76. {
  77. return CreateHeader (name, ns, value, new DataContractSerializer (value.GetType ()),
  78. must_understand, actor, relay);
  79. }
  80. public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter,
  81. bool must_understand, string actor)
  82. {
  83. return CreateHeader (name, ns, value, formatter, must_understand, actor, default_relay);
  84. }
  85. public static MessageHeader CreateHeader (string name, string ns, object value, XmlObjectSerializer formatter,
  86. bool must_understand, string actor, bool relay)
  87. {
  88. // FIXME: how to get IsReferenceParameter ?
  89. return new DefaultMessageHeader (name, ns, value, formatter, default_is_ref, must_understand, actor, relay);
  90. }
  91. public virtual bool IsMessageVersionSupported (MessageVersion version)
  92. {
  93. if (version.Envelope == EnvelopeVersion.Soap12)
  94. if (Actor == EnvelopeVersion.Soap11.NextDestinationActorValue)
  95. return false;
  96. if (version.Envelope == EnvelopeVersion.Soap11)
  97. if (Actor == EnvelopeVersion.Soap12.NextDestinationActorValue ||
  98. Actor == EnvelopeVersion.Soap12UltimateReceiver)
  99. return false;
  100. // by default, it's always supported
  101. return true;
  102. }
  103. protected abstract void OnWriteHeaderContents (XmlDictionaryWriter writer, MessageVersion version);
  104. protected virtual void OnWriteStartHeader (XmlDictionaryWriter writer, MessageVersion version)
  105. {
  106. var dic = Constants.SoapDictionary;
  107. XmlDictionaryString name, ns;
  108. var prefix = Prefix ?? (Namespace.Length > 0 ? writer.LookupPrefix (Namespace) : String.Empty);
  109. if (dic.TryLookup (Name, out name) && dic.TryLookup (Namespace, out ns))
  110. writer.WriteStartElement (prefix, name, ns);
  111. else
  112. writer.WriteStartElement (prefix, this.Name, this.Namespace);
  113. WriteHeaderAttributes (writer, version);
  114. }
  115. public override string ToString ()
  116. {
  117. StringBuilder sb = new StringBuilder ();
  118. XmlWriter w = XmlWriter.Create (sb, writer_settings);
  119. WriteHeader (w, MessageVersion.Default);
  120. w.Close ();
  121. return sb.ToString ();
  122. }
  123. public void WriteHeader (XmlDictionaryWriter writer, MessageVersion version)
  124. {
  125. if (writer == null)
  126. throw new ArgumentNullException ("writer is null.");
  127. if (version == null)
  128. throw new ArgumentNullException ("version is null.");
  129. if (version.Envelope == EnvelopeVersion.None)
  130. return;
  131. WriteStartHeader (writer, version);
  132. WriteHeaderContents (writer, version);
  133. writer.WriteEndElement ();
  134. }
  135. public void WriteHeader (XmlWriter writer, MessageVersion version)
  136. {
  137. WriteHeader (XmlDictionaryWriter.CreateDictionaryWriter (writer), version);
  138. }
  139. protected void WriteHeaderAttributes (XmlDictionaryWriter writer, MessageVersion version)
  140. {
  141. var dic = Constants.SoapDictionary;
  142. if (Id != null)
  143. writer.WriteAttributeString ("u", dic.Add ("Id"), dic.Add (Constants.WsuNamespace), Id);
  144. if (!String.IsNullOrEmpty (Actor)) {
  145. if (version.Envelope == EnvelopeVersion.Soap11)
  146. writer.WriteAttributeString ("s", dic.Add ("actor"), dic.Add (version.Envelope.Namespace), Actor);
  147. if (version.Envelope == EnvelopeVersion.Soap12)
  148. writer.WriteAttributeString ("s", dic.Add ("role"), dic.Add (version.Envelope.Namespace), Actor);
  149. }
  150. // mustUnderstand is the same across SOAP 1.1 and 1.2
  151. if (MustUnderstand == true)
  152. writer.WriteAttributeString ("s", dic.Add ("mustUnderstand"), dic.Add (version.Envelope.Namespace), "1");
  153. // relay is only available on SOAP 1.2
  154. if (Relay == true && version.Envelope == EnvelopeVersion.Soap12)
  155. writer.WriteAttributeString ("s", dic.Add ("relay"), dic.Add (version.Envelope.Namespace), "true");
  156. }
  157. public void WriteHeaderContents (XmlDictionaryWriter writer, MessageVersion version)
  158. {
  159. this.OnWriteHeaderContents (writer, version);
  160. }
  161. public void WriteStartHeader (XmlDictionaryWriter writer, MessageVersion version)
  162. {
  163. this.OnWriteStartHeader (writer, version);
  164. }
  165. public override string Actor { get { return default_actor; }}
  166. public override bool IsReferenceParameter { get { return default_is_ref; }}
  167. public override bool MustUnderstand { get { return default_must_understand; }}
  168. public override bool Relay { get { return default_relay; }}
  169. internal class XmlMessageHeader : MessageHeader
  170. {
  171. bool is_ref, must_understand, relay;
  172. string actor;
  173. #if MOBILE
  174. string body;
  175. #else
  176. // This is required to completely clone body xml that
  177. // does not introduce additional xmlns declarations that
  178. // blocks canonicalized copy of the input XML.
  179. XmlDocument body;
  180. #endif
  181. string local_name;
  182. string namespace_uri;
  183. public XmlMessageHeader (XmlReader reader, MessageVersion version)
  184. {
  185. var soapNS = version.Envelope.Namespace;
  186. var addrNS = version.Addressing.Namespace;
  187. Prefix = reader.Prefix;
  188. Id = reader.GetAttribute ("Id", Constants.WsuNamespace);
  189. string s = reader.GetAttribute ("relay", soapNS);
  190. relay = s != null ? XmlConvert.ToBoolean (s) : false;
  191. s = reader.GetAttribute ("mustUnderstand", soapNS);
  192. must_understand = s != null ? XmlConvert.ToBoolean (s) : false;
  193. actor = reader.GetAttribute ("actor", soapNS) ?? String.Empty;
  194. s = reader.GetAttribute ("IsReferenceParameter", addrNS);
  195. is_ref = s != null ? XmlConvert.ToBoolean (s) : false;
  196. local_name = reader.LocalName;
  197. namespace_uri = reader.NamespaceURI;
  198. #if MOBILE
  199. body = reader.ReadOuterXml ();
  200. #else
  201. body = new XmlDocument ();
  202. var w = body.CreateNavigator ().AppendChild ();
  203. w.WriteNode (reader, false);
  204. w.Close ();
  205. #endif
  206. }
  207. public XmlReader CreateReader ()
  208. {
  209. #if MOBILE
  210. var reader = XmlReader.Create (new StringReader (body));
  211. #else
  212. var reader = new XmlNodeReader (body);
  213. #endif
  214. reader.MoveToContent ();
  215. return reader;
  216. }
  217. protected override void OnWriteHeaderContents (
  218. XmlDictionaryWriter writer, MessageVersion version)
  219. {
  220. var r = CreateReader ();
  221. r.MoveToContent ();
  222. if (r.IsEmptyElement)
  223. return; // write nothing
  224. for (r.Read (); r.NodeType != XmlNodeType.EndElement;)
  225. writer.WriteNode (r, false);
  226. }
  227. public override string Actor { get { return actor; }}
  228. public override bool IsReferenceParameter { get { return is_ref; }}
  229. public override bool MustUnderstand { get { return must_understand; }}
  230. public override string Name { get { return local_name; }}
  231. public override string Namespace { get { return namespace_uri; }}
  232. public override bool Relay { get { return relay; }}
  233. }
  234. internal class DefaultMessageHeader : MessageHeader
  235. {
  236. string actor, name, ns;
  237. object value;
  238. XmlObjectSerializer formatter;
  239. bool is_ref, must_understand, relay;
  240. internal DefaultMessageHeader (string name, string ns, object value, XmlObjectSerializer formatter,
  241. bool isReferenceParameter,
  242. bool mustUnderstand, string actor, bool relay)
  243. {
  244. this.name = name;
  245. this.ns = ns;
  246. this.value = value;
  247. this.formatter = formatter;
  248. this.is_ref = isReferenceParameter;
  249. this.must_understand = mustUnderstand;
  250. this.actor = actor ?? String.Empty;
  251. this.relay = relay;
  252. }
  253. protected override void OnWriteHeaderContents (XmlDictionaryWriter writer,
  254. MessageVersion version)
  255. {
  256. // FIXME: it's a nasty workaround just to avoid UniqueId output as a string, for bug #577139.
  257. if (Value is UniqueId)
  258. writer.WriteValue ((UniqueId) Value);
  259. else
  260. this.formatter.WriteObjectContent (writer, value);
  261. }
  262. public object Value { get { return value; } }
  263. public override string Actor { get { return actor; }}
  264. public override bool IsReferenceParameter { get { return is_ref; }}
  265. public override bool MustUnderstand { get { return must_understand; }}
  266. public override string Name { get { return name; }}
  267. public override string Namespace { get { return ns; }}
  268. public override bool Relay { get { return relay; }}
  269. }
  270. }
  271. }