程式設計 WCF 安全性
本主題描述用來建立安全 Windows Communication Foundation (WCF) 應用程式的基本程序設計工作。 本主題僅涵蓋驗證、機密性和完整性,統稱為 傳輸安全性。 本主題未涵蓋授權(對資源或服務的存取控制):如需授權的相關信息,請參閱 授權。
注意
如需了解安全性概念的寶貴簡介,尤其是關於 WCF,請參閱 MSDN 上的一組作法和模式教學課程,內容包括 Web 服務增強功能(WSE)3.0 的案例、模式和實作指引。
程序設計 WCF 安全性是以下列三個步驟為基礎:安全性模式、用戶端認證類型和認證值。 您可以透過程式代碼或設定來執行這些步驟。
下列說明在 WCF 中使用安全性模式進行程式設計的一般步驟:
選取適合您應用程式需求的其中一個預先定義系結。 如需系結選項的清單,請參閱 System-Provided 系結。 根據預設,幾乎每個系結都已啟用安全性。 其中一個例外狀況是 BasicHttpBinding 類別(使用組態、<basicHttpBinding>)。
您選取的系結會決定傳輸。 例如,WSHttpBinding 使用 HTTP 作為傳輸;NetTcpBinding 使用 TCP。
選取系結的其中一種安全性模式。 請注意,您選取的系結會決定可用的模式選擇。 例如,WSDualHttpBinding 不允許傳輸安全性(它不是選項)。 同樣地,MsmqIntegrationBinding 和 NetNamedPipeBinding 都不允許訊息安全性。
您有下列三個選擇:
Transport
傳輸安全性取決於您所選系結所使用的機制。 例如,如果您使用
WSHttpBinding
則安全性機制是安全套接字層 (SSL) (也是 HTTPS 通訊協定的機制)。 一般而言,傳輸安全性的主要優點是,不論您使用哪一種傳輸,它都提供良好的輸送量。 不過,它確實有兩個限制:第一個是傳輸機制會指定用來驗證使用者的認證類型。 只有當服務需要與其他需要不同類型的認證的服務互作時,才會有缺點。 第二個原因是,由於安全性未在訊息層級套用,因此安全性會以逐躍方式實作,而不是端對端實作。 只有在用戶端和服務之間的訊息路徑包含中繼時,這個後一項限制才會發生問題。 如需更多有關應用哪種傳輸方式的詳細資訊,請參閱 選擇傳輸方式。 如需使用傳輸安全性的詳細資訊,請參閱 傳輸安全性概觀。Message
訊息安全性表示每個訊息都包含必要的標頭和數據,以保護訊息安全。 因為標頭的組成會有所不同,因此您可以包含任意數目的認證。 如果您與其他要求傳輸機制無法提供的特定認證類型的服務互通,或訊息必須與多個服務搭配使用,則這會成為一個因素,其中每個服務需要不同的認證類型。
如需詳細資訊,請參閱 訊息安全性。
TransportWithMessageCredential
此選項會使用傳輸層來保護訊息傳輸,而每個訊息都包含其他服務所需的豐富認證。 這結合了傳輸安全性的效能優勢與訊息安全性的豐富認證優勢。 這適用於下列系結:BasicHttpBinding、WSFederationHttpBinding、NetPeerTcpBinding和 WSHttpBinding。
如果您決定使用 HTTP 的傳輸安全性(換句話說為 HTTPS),您也必須使用 SSL 憑證來設定主機,並在埠上啟用 SSL。 如需詳細資訊,請參閱 HTTP 傳輸安全性。
如果您使用 WSHttpBinding,而且不需要建立安全工作階段,請將 EstablishSecurityContext 屬性設定為
false
。當客戶端和服務使用對稱密鑰建立通道時,就會發生安全會話(用戶端和伺服器在對話關閉前使用相同的密鑰長度)。
視需要選取客戶端認證類型。 如需詳細資訊,請參閱 選取認證類型。 下列客戶端認證類型可供使用:
Windows
Certificate
Digest
Basic
UserName
NTLM
IssuedToken
根據設定模式的方式,您必須設定認證類型。 例如,如果您已選取 wsHttpBinding
,並將模式設定為 「Message」,則您也可以將 Message 元素的 clientCredentialType
屬性設定為下列其中一個值:None
、Windows
、UserName
、Certificate
和 IssuedToken
,如下列組態範例所示。
<system.serviceModel>
<bindings>
<wsHttpBinding>
<binding name="myBinding">
<security mode="Message"/>
<message clientCredentialType="Windows"/>
</binding>
</wsHttpBinding>
</bindings>
</system.serviceModel>
或在程式代碼中:
WSHttpBinding b = new WSHttpBinding();
b.Name = "myBinding";
b.Security.Mode = SecurityMode.Message;
b.Security.Message.ClientCredentialType=MessageCredentialType.Windows;
Dim b As New WSHttpBinding()
b.Name = "myBinding"
b.Security.Mode = SecurityMode.Message
b.Security.Message.ClientCredentialType = MessageCredentialType.Windows
選取客戶端認證類型之後,您必須設定服務與用戶端要使用的實際認證。 在服務上,認證是使用 ServiceCredentials 類別來設定,並由 ServiceHostBase 類別的 Credentials 屬性傳回。 使用中的系結表示服務認證類型、所選擇的安全性模式,以及客戶端認證的類型。 下列程式代碼會設定服務認證的憑證。
// Create the binding for an endpoint.
NetTcpBinding b = new NetTcpBinding();
b.Security.Mode = SecurityMode.Message;
// Create the ServiceHost for a calculator.
Uri baseUri = new Uri("net.tcp://MachineName/tcpBase");
Uri[] baseAddresses = new Uri[] { baseUri };
ServiceHost sh = new ServiceHost(typeof(Calculator), baseAddresses);
// Add an endpoint using the binding and a new address.
Type c = typeof(ICalculator);
sh.AddServiceEndpoint(c, b, "MyEndpoint");
// Set a certificate as the credential for the service.
sh.Credentials.ServiceCertificate.SetCertificate(
StoreLocation.LocalMachine,
StoreName.My,
X509FindType.FindBySubjectName,
"client.com");
try
{
sh.Open();
Console.WriteLine("Listening....");
Console.ReadLine();
sh.Close();
}
catch (CommunicationException ce)
{
Console.WriteLine($"A communication error occurred: {ce.Message}");
Console.WriteLine();
}
catch (System.Exception exc)
{
Console.WriteLine($"An unforeseen error occurred: {exc.Message}");
Console.ReadLine();
}
' Create the binding for an endpoint.
Dim b As New NetTcpBinding()
b.Security.Mode = SecurityMode.Message
' Create the ServiceHost for a calculator.
Dim baseUri As New Uri("net.tcp://MachineName/tcpBase")
Dim baseAddresses() As Uri = {baseUri}
Dim sh As New ServiceHost(GetType(Calculator), baseAddresses)
' Add an endpoint using the binding and a new address.
Dim c As Type = GetType(ICalculator)
sh.AddServiceEndpoint(c, b, "MyEndpoint")
' Set a certificate as the credential for the service.
sh.Credentials.ServiceCertificate.SetCertificate( _
StoreLocation.LocalMachine, _
StoreName.My, _
X509FindType.FindBySubjectName, _
"contoso.com")
Try
sh.Open()
Console.WriteLine("Listening....")
Console.ReadLine()
sh.Close()
Catch ce As CommunicationException
Console.WriteLine("A communication error occurred: {0}", ce.Message)
Console.WriteLine()
Catch exc As System.Exception
Console.WriteLine("An unforeseen error occurred: {0}", exc.Message)
Console.ReadLine()
End Try
在用戶端上,使用 ClientCredentials 類別設定客戶端認證值,並由 ClientBase<TChannel> 類別的 ClientCredentials 屬性傳回。 下列程式代碼會使用 TCP 通訊協定,將憑證設定為用戶端上的認證。
// Create a NetTcpBinding and set its security properties. The
// security mode is Message, and the client must be authenticated with
// Windows. Therefore the client must be on the same Windows domain.
NetTcpBinding b = new NetTcpBinding();
b.Security.Mode = SecurityMode.Message;
b.Security.Message.ClientCredentialType = MessageCredentialType.Windows;
// Set a Type variable for use when constructing the endpoint.
Type c = typeof(ICalculator);
// Create a base address for the service.
Uri tcpBaseAddress =
new Uri("net.tcp://machineName.Domain.Contoso.com:8036/serviceName");
// The base address is in an array of URI objects.
Uri[] baseAddresses = new Uri[] { tcpBaseAddress };
// Create the ServiceHost with type and base addresses.
ServiceHost sh = new ServiceHost(typeof(CalculatorClient), baseAddresses);
// Add an endpoint to the service using the service type and binding.
sh.AddServiceEndpoint(c, b, "");
sh.Open();
string address = sh.Description.Endpoints[0].ListenUri.AbsoluteUri;
Console.WriteLine($"Listening @ {address}");
Console.WriteLine("Press enter to close the service");
Console.ReadLine();
' Create a NetTcpBinding and set its security properties. The
' security mode is Message, and the client must be authenticated with
' Windows. Therefore the client must be on the same Windows domain.
Dim b As New NetTcpBinding()
b.Security.Mode = SecurityMode.Message
b.Security.Message.ClientCredentialType = MessageCredentialType.Windows
' Set a Type variable for use when constructing the endpoint.
Dim c As Type = GetType(ICalculator)
' Create a base address for the service.
Dim tcpBaseAddress As New Uri("net.tcp://machineName.Domain.Contoso.com:8036/serviceName")
' The base address is in an array of URI objects.
Dim baseAddresses() As Uri = {tcpBaseAddress}
' Create the ServiceHost with type and base addresses.
Dim sh As New ServiceHost(GetType(CalculatorClient), baseAddresses)
' Add an endpoint to the service using the service type and binding.
sh.AddServiceEndpoint(c, b, "")
sh.Open()
Dim address As String = sh.Description.Endpoints(0).ListenUri.AbsoluteUri
Console.WriteLine("Listening @ {0}", address)
Console.WriteLine("Press enter to close the service")
Console.ReadLine()