WSFederationHttpBinding.cs 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. //------------------------------------------------------------
  2. // Copyright (c) Microsoft Corporation. All rights reserved.
  3. //------------------------------------------------------------
  4. namespace System.ServiceModel
  5. {
  6. using System;
  7. using System.Text;
  8. using System.Collections.Generic;
  9. using System.Collections.ObjectModel;
  10. using System.Configuration;
  11. using System.Globalization;
  12. using System.Net;
  13. using System.Net.Security;
  14. using System.Runtime.Serialization;
  15. using System.Security.Principal;
  16. using System.ServiceModel.Channels;
  17. using System.ServiceModel.Configuration;
  18. using System.ServiceModel.Security;
  19. using System.Xml;
  20. using System.ComponentModel;
  21. public class WSFederationHttpBinding : WSHttpBindingBase
  22. {
  23. static readonly MessageSecurityVersion WSMessageSecurityVersion = MessageSecurityVersion.WSSecurity11WSTrustFebruary2005WSSecureConversationFebruary2005WSSecurityPolicy11BasicSecurityProfile10;
  24. Uri privacyNoticeAt;
  25. int privacyNoticeVersion;
  26. WSFederationHttpSecurity security = new WSFederationHttpSecurity();
  27. public WSFederationHttpBinding(string configName)
  28. : this()
  29. {
  30. ApplyConfiguration(configName);
  31. }
  32. public WSFederationHttpBinding()
  33. : base()
  34. {
  35. }
  36. public WSFederationHttpBinding(WSFederationHttpSecurityMode securityMode)
  37. : this(securityMode, false)
  38. {
  39. }
  40. public WSFederationHttpBinding(WSFederationHttpSecurityMode securityMode, bool reliableSessionEnabled)
  41. : base(reliableSessionEnabled)
  42. {
  43. security.Mode = securityMode;
  44. }
  45. internal WSFederationHttpBinding(WSFederationHttpSecurity security, PrivacyNoticeBindingElement privacy, bool reliableSessionEnabled)
  46. : base(reliableSessionEnabled)
  47. {
  48. this.security = security;
  49. if (null != privacy)
  50. {
  51. this.privacyNoticeAt = privacy.Url;
  52. this.privacyNoticeVersion = privacy.Version;
  53. }
  54. }
  55. [DefaultValue(null)]
  56. public Uri PrivacyNoticeAt
  57. {
  58. get { return this.privacyNoticeAt; }
  59. set { this.privacyNoticeAt = value; }
  60. }
  61. [DefaultValue(0)]
  62. public int PrivacyNoticeVersion
  63. {
  64. get { return this.privacyNoticeVersion; }
  65. set { this.privacyNoticeVersion = value; }
  66. }
  67. public WSFederationHttpSecurity Security
  68. {
  69. get { return this.security; }
  70. set
  71. {
  72. if (value == null)
  73. throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("value");
  74. this.security = value;
  75. }
  76. }
  77. void ApplyConfiguration(string configurationName)
  78. {
  79. WSFederationHttpBindingCollectionElement section = WSFederationHttpBindingCollectionElement.GetBindingCollectionElement();
  80. WSFederationHttpBindingElement element = section.Bindings[configurationName];
  81. if (element == null)
  82. {
  83. throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ConfigurationErrorsException(
  84. SR.GetString(SR.ConfigInvalidBindingConfigurationName,
  85. configurationName,
  86. ConfigurationStrings.WSFederationHttpBindingCollectionElementName)));
  87. }
  88. else
  89. {
  90. element.ApplyConfiguration(this);
  91. }
  92. }
  93. PrivacyNoticeBindingElement CreatePrivacyPolicy()
  94. {
  95. PrivacyNoticeBindingElement privacy = null;
  96. if (this.PrivacyNoticeAt != null)
  97. {
  98. privacy = new PrivacyNoticeBindingElement();
  99. privacy.Url = this.PrivacyNoticeAt;
  100. privacy.Version = this.privacyNoticeVersion;
  101. }
  102. return privacy;
  103. }
  104. // if you make changes here, see also WS2007FederationHttpBinding.TryCreate()
  105. internal static bool TryCreate(SecurityBindingElement sbe, TransportBindingElement transport, PrivacyNoticeBindingElement privacy, ReliableSessionBindingElement rsbe, TransactionFlowBindingElement tfbe, out Binding binding)
  106. {
  107. bool isReliableSession = (rsbe != null);
  108. binding = null;
  109. // reverse GetTransport
  110. HttpTransportSecurity transportSecurity = new HttpTransportSecurity();
  111. WSFederationHttpSecurityMode mode;
  112. if (!GetSecurityModeFromTransport(transport, transportSecurity, out mode))
  113. {
  114. return false;
  115. }
  116. HttpsTransportBindingElement httpsBinding = transport as HttpsTransportBindingElement;
  117. if (httpsBinding != null && httpsBinding.MessageSecurityVersion != null)
  118. {
  119. if (httpsBinding.MessageSecurityVersion.SecurityPolicyVersion != WSMessageSecurityVersion.SecurityPolicyVersion)
  120. {
  121. return false;
  122. }
  123. }
  124. WSFederationHttpSecurity security;
  125. if (TryCreateSecurity(sbe, mode, transportSecurity, isReliableSession, out security))
  126. {
  127. binding = new WSFederationHttpBinding(security, privacy, isReliableSession);
  128. }
  129. if (rsbe != null && rsbe.ReliableMessagingVersion != ReliableMessagingVersion.WSReliableMessagingFebruary2005)
  130. {
  131. return false;
  132. }
  133. if (tfbe != null && tfbe.TransactionProtocol != TransactionProtocol.WSAtomicTransactionOctober2004)
  134. {
  135. return false;
  136. }
  137. return binding != null;
  138. }
  139. protected override TransportBindingElement GetTransport()
  140. {
  141. if (security.Mode == WSFederationHttpSecurityMode.None || security.Mode == WSFederationHttpSecurityMode.Message)
  142. {
  143. return this.HttpTransport;
  144. }
  145. else
  146. {
  147. return this.HttpsTransport;
  148. }
  149. }
  150. internal static bool GetSecurityModeFromTransport(TransportBindingElement transport, HttpTransportSecurity transportSecurity, out WSFederationHttpSecurityMode mode)
  151. {
  152. mode = WSFederationHttpSecurityMode.None | WSFederationHttpSecurityMode.Message | WSFederationHttpSecurityMode.TransportWithMessageCredential;
  153. if (transport is HttpsTransportBindingElement)
  154. {
  155. mode = WSFederationHttpSecurityMode.TransportWithMessageCredential;
  156. }
  157. else if (transport is HttpTransportBindingElement)
  158. {
  159. mode = WSFederationHttpSecurityMode.None | WSFederationHttpSecurityMode.Message;
  160. }
  161. else
  162. {
  163. return false;
  164. }
  165. return true;
  166. }
  167. protected override SecurityBindingElement CreateMessageSecurity()
  168. {
  169. return security.CreateMessageSecurity(this.ReliableSession.Enabled, WSMessageSecurityVersion);
  170. }
  171. // if you make changes here, see also WS2007FederationHttpBinding.TryCreateSecurity()
  172. static bool TryCreateSecurity(SecurityBindingElement sbe, WSFederationHttpSecurityMode mode, HttpTransportSecurity transportSecurity, bool isReliableSession, out WSFederationHttpSecurity security)
  173. {
  174. if (!WSFederationHttpSecurity.TryCreate(sbe, mode, transportSecurity, isReliableSession, WSMessageSecurityVersion, out security))
  175. return false;
  176. // the last check: make sure that security binding element match the incoming security
  177. return SecurityElement.AreBindingsMatching(security.CreateMessageSecurity(isReliableSession, WSMessageSecurityVersion), sbe);
  178. }
  179. public override BindingElementCollection CreateBindingElements()
  180. { // return collection of BindingElements
  181. BindingElementCollection bindingElements = base.CreateBindingElements();
  182. // order of BindingElements is important
  183. PrivacyNoticeBindingElement privacy = this.CreatePrivacyPolicy();
  184. if (privacy != null)
  185. {
  186. // This must go first.
  187. bindingElements.Insert(0, privacy);
  188. }
  189. return bindingElements;
  190. }
  191. [EditorBrowsable(EditorBrowsableState.Never)]
  192. public bool ShouldSerializeSecurity()
  193. {
  194. return this.Security.InternalShouldSerialize();
  195. }
  196. }
  197. }