繁体   English   中英

Java 应用程序中的 PKIX 路径构建失败

[英]PKIX path building failed in Java application

在将我的应用程序从 Windows 2000 移动到 Windows 2008 R2 Server 之后,我一直在努力让我的应用程序运行近一个星期。

程序,流程:

  1. 已安装 Java JDK 1.7.0_25
  2. 设置系统环境变量JAVA_HOMEC:\\Progra~1\\Java\\jdk1.7.0_25\\
  3. 使用keytool将证书导入 cacerts
  4. 使用-list确保证书存在于keytool

我试图用InstallCert重复步骤 3 ,以确保我没有搞砸任何事情。

上述方法没有解决我的问题,所以我尝试以编程方式进行:

System.setProperty("javax.net.ssl.trustStore",
"C:/Progra~1/Java/jdk1.7.0_25/jre/lib/security/cacerts");
System.setProperty("javax.net.ssl.trustStorePassword", "changeit");

仍然没有任何运气。 我被卡住了,不太确定从这里往哪个方向。

堆栈跟踪:

javax.net.ssl.SSLHandshakeException: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target
    at sun.security.ssl.Alerts.getSSLException(Alerts.java:192)
    at sun.security.ssl.SSLSocketImpl.fatal(SSLSocketImpl.java:1886)
    at sun.security.ssl.Handshaker.fatalSE(Handshaker.java:276)
    at sun.security.ssl.Handshaker.fatalSE(Handshaker.java:270)
    at sun.security.ssl.ClientHandshaker.serverCertificate(ClientHandshaker.java:1341)
    at sun.security.ssl.ClientHandshaker.processMessage(ClientHandshaker.java:153)
    at sun.security.ssl.Handshaker.processLoop(Handshaker.java:868)
    at sun.security.ssl.Handshaker.process_record(Handshaker.java:804)
    at sun.security.ssl.SSLSocketImpl.readRecord(SSLSocketImpl.java:1016)
    at sun.security.ssl.SSLSocketImpl.performInitialHandshake(SSLSocketImpl.java:1312)
    at sun.security.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1339)
    at sun.security.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1323)
    at sun.net.www.protocol.https.HttpsClient.afterConnect(HttpsClient.java:515)
    at sun.net.www.protocol.https.AbstractDelegateHttpsURLConnection.connect(AbstractDelegateHttpsURLConnection.java:185)
    at sun.net.www.protocol.https.HttpsURLConnectionImpl.connect(HttpsURLConnectionImpl.java:153)
    at util.SMS.send(SMS.java:93)
    at domain.ActivationSMSSenderMain.sendActivationMessagesToCustomers(ActivationSMSSenderMain.java:80)
    at domain.ActivationSMSSenderMain.<init>(ActivationSMSSenderMain.java:44)
    at domain.ActivationSMSSenderMain.main(ActivationSMSSenderMain.java:341)
Caused by: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target
    at sun.security.validator.PKIXValidator.doBuild(PKIXValidator.java:385)
    at sun.security.validator.PKIXValidator.engineValidate(PKIXValidator.java:292)
    at sun.security.validator.Validator.validate(Validator.java:260)
    at sun.security.ssl.X509TrustManagerImpl.validate(X509TrustManagerImpl.java:326)
    at sun.security.ssl.X509TrustManagerImpl.checkTrusted(X509TrustManagerImpl.java:231)
    at sun.security.ssl.X509TrustManagerImpl.checkServerTrusted(X509TrustManagerImpl.java:126)
    at sun.security.ssl.ClientHandshaker.serverCertificate(ClientHandshaker.java:1323)
    ... 14 more
Caused by: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target
    at sun.security.provider.certpath.SunCertPathBuilder.engineBuild(SunCertPathBuilder.java:196)
    at java.security.cert.CertPathBuilder.build(CertPathBuilder.java:268)
    at sun.security.validator.PKIXValidator.doBuild(PKIXValidator.java:380)
    ... 20 more

更新:

System.out.println(System.getProperty("javax.net.ssl.trustStore")); System.out.println(System.getProperty("javax.net.ssl.keyStore"));

返回null

我遇到了类似的问题,其原因和解决方案都非常简单:

主要原因:没有使用 keytool 导入正确的证书

注意:仅导入根 CA(或您自己的自签名)证书

注意:不要导入中间的、非证书链根证书

imap.gmail.com 的解决方案示例

  1. 确定根 CA 证书:

     openssl s_client -showcerts -connect imap.gmail.com:993

    在这种情况下,我们发现根 CA 是Equifax Secure Certificate Authority

  2. 下载根 CA 证书
  3. 通过与此处找到的信息进行比较,验证下载的证书是否具有正确的 SHA-1 和/或 MD5 指纹
  4. javax.net.ssl.trustStore导入证书:

     keytool -import -alias gmail_imap -file Equifax_Secure_Certificate_Authority.pem
  5. 运行你的java代码

您已将证书导入到 JDK 中提供的 JRE 的信任库中,但您正在运行直接安装的 JRE 的 java.exe。

编辑

为了清楚起见,并解决下面评论中的误解,您需要将证书导入到您打算使用的 JREcacerts文件中,并且很少会在 JDK 内部传送该证书,因为客户端通常不会有 JDK。 以下评论中任何暗示其他内容的内容都应该被忽略,因为这里没有表达我的意图。

更好的解决方案是创建您自己的信任库,从cacerts文件的副本开始,并通过系统属性javax.net.ssl.trustStore.专门告诉 Java 使用该信任库javax.net.ssl.trustStore.

您应该将构建这部分作为构建过程的一部分,以便与 JDK 升级引起的cacerts文件的更改保持cacerts

如果您使用的是 Eclipse,只需在 Eclipse Windows--> 首选项--> java ---> 安装的 JRE 中交叉检查当前 JRE 和您配置证书的 JRE。 如果没有删除 JRE 并在安装证书的位置添加 jre

根据您的 pastebin,您需要将proxy.tkk.com证书添加到信任库。

在 Windows 上,您可以尝试以下步骤:

  1. 从网站下载根 CA 证书。
  2. 使用 JRE 在目录/lib/security找到一个文件 jssecacerts(您可以使用命令System.out.println(System.getProperty("java.home");找到包含当前 JRE 的文件夹)。备份文件。
  3. 下载程序portecle
  4. 在 portecle 中打开 jssecacerts 文件。
  5. 输入密码:changeit。
  6. 使用 porticle 导入下载的证书(工具 > 导入可信证书)。
  7. 单击保存。
  8. 替换原来的文件jssecacerts。

在我的情况下,问题是通过安装Oracle 的官方 JDK 10而不是使用我的 Ubuntu 附带的默认 OpenJDK 解决的。 这是我遵循的指南: https : //www.linuxuprising.com/2018/04/install-oracle-java-10-in-ubuntu-or.html

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM