SSL 流身份验证失败

本文关键字:失败 身份验证 SSL | 更新日期: 2023-09-27 18:34:16

大家,我正在尝试写一些关于SSL的东西,这是一个问题:

我在下面构建了东西:

  • CA证书(自制CA)
  • 服务器
  • pfx、服务器证书、服务器密钥(由自制 CA 签名为"本地主机")

现在我正在使用.Net SslStream来测试连接:(客户端和服务器在不同的线程中,并且TCP连接已经建立)

客户:

sslStream.AuthenticateAsClient("localhost");

服务器:

sslStream.AuthenticateAsServer(serverCert); 
//serverCert is X509Certificate2 built from "server.pfx"
客户端

的身份验证为客户端方法将引发异常

"根据验证过程,远程证书无效。"

我想原因是服务器的证书是由不受信任的CA签名的,因此身份验证失败,那么如何将CA证书添加到我的信任列表中?

我尝试在客户端代码中添加下面的代码,但它不起作用

        X509Store store = new X509Store(StoreName.TrustedPublisher, StoreLocation.CurrentUser);
        store.Open(OpenFlags.ReadWrite);
        store.Add(new X509Certificate2(Resources.CACertPath));
        store.Close();
        sslStream.AuthenticateAsClient("localhost");

SSL 流身份验证失败

以下代码将避免 Windows 证书存储并验证链。

我认为没有理由将验证链所需的 CA 证书添加到证书存储中的数百个证书中。这意味着Windows将尝试使用"数百+ 1"证书而不是真正需要的证书来验证链。

还没有弄清楚默认情况下如何使用这个链(chain2下面),这样就不需要回调了。也就是说,将其安装在 ssl 套接字上,连接将"正常工作"。而且我还没有弄清楚如何安装它,使其传递到回调中。也就是说,我必须为每次调用回调构建链。我认为这些是 .Net 中的体系结构缺陷,但我可能错过了一些明显的东西。

函数的名称无关紧要。下面,VerifyServerCertificate 是与类中的RemoteCertificateValidationCallback相同的回调SslStream。您也可以将其用于ServicePointManager中的ServerCertificateValidationCallback

static bool VerifyServerCertificate(object sender, X509Certificate certificate,
    X509Chain chain, SslPolicyErrors sslPolicyErrors)
{
    try
    {
        String CA_FILE = "ca-cert.der";
        X509Certificate2 ca = new X509Certificate2(CA_FILE);
        X509Chain chain2 = new X509Chain();
        chain2.ChainPolicy.ExtraStore.Add(ca);
        // Check all properties
        chain2.ChainPolicy.VerificationFlags = X509VerificationFlags.NoFlag;
        // This setup does not have revocation information
        chain2.ChainPolicy.RevocationMode = X509RevocationMode.NoCheck;
        // Build the chain
        chain2.Build(new X509Certificate2(certificate));
        // Are there any failures from building the chain?
        if (chain2.ChainStatus.Length == 0)
            return true;
        // If there is a status, verify the status is NoError
        bool result = chain2.ChainStatus[0].Status == X509ChainStatusFlags.NoError;
        Debug.Assert(result == true);
        return result;
    }
    catch (Exception ex)
    {
        Console.WriteLine(ex);
    }
    return false;
}