为什么我在处理PayPal付款时突然收到"PKIX 路径构建失败"错误



By基于Java的网站使用Paypal处理支付。该错误始于昨晚格林尼治标准时间晚上11点左右,没有代码更改,我无法确定问题是否出在我的托管提供商或Paypla身上。

引发异常的代码是

                // Send back to PayPal system to validate using POST
                URL u = new URL(URL_PAYPAL_VALIDATE);
                HttpURLConnection uc = (HttpURLConnection) u.openConnection();
                uc.setDoOutput(true);
                uc.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");
                PrintWriter pw = new PrintWriter(uc.getOutputStream());
                pw.println(str);
                pw.close();

并发生在uc.getOuptStream()上

这是的例外

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 com.sun.net.ssl.internal.ssl.Alerts.getSSLException(Alerts.java:174)
        at com.sun.net.ssl.internal.ssl.SSLSocketImpl.fatal(SSLSocketImpl.java:1591)
        at com.sun.net.ssl.internal.ssl.Handshaker.fatalSE(Handshaker.java:187)
        at com.sun.net.ssl.internal.ssl.Handshaker.fatalSE(Handshaker.java:181)
        at com.sun.net.ssl.internal.ssl.ClientHandshaker.serverCertificate(ClientHandshaker.java:975)
        at com.sun.net.ssl.internal.ssl.ClientHandshaker.processMessage(ClientHandshaker.java:123)
        at com.sun.net.ssl.internal.ssl.Handshaker.processLoop(Handshaker.java:516)
        at com.sun.net.ssl.internal.ssl.Handshaker.process_record(Handshaker.java:454)
        at com.sun.net.ssl.internal.ssl.SSLSocketImpl.readRecord(SSLSocketImpl.java:884)
        at com.sun.net.ssl.internal.ssl.SSLSocketImpl.performInitialHandshake(SSLSocketImpl.java:1096)
        at com.sun.net.ssl.internal.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1123)
        at com.sun.net.ssl.internal.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1107)
        at sun.net.www.protocol.https.HttpsClient.afterConnect(HttpsClient.java:405)
        at sun.net.www.protocol.https.AbstractDelegateHttpsURLConnection.connect(AbstractDelegateHttpsURLConnection.java:166)
        at sun.net.www.protocol.http.HttpURLConnection.getOutputStream(HttpURLConnection.java:832)
        at sun.net.www.protocol.https.HttpsURLConnectionImpl.getOutputStream(HttpsURLConnectionImpl.java:230)
        at com.jthink.store.action.VerifyPayment.handleRequest(VerifyPayment.java:160)
        at com.jthink.store.JThinkStoreServlet.doGet(JThinkStoreServlet.java:45)
        at javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
        at javax.servlet.http.HttpServlet.service(HttpServlet.java:722)
        at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:305)
        at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
        at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:222)
        at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:123)
        at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:472)
        at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:168)
        at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:99)
        at org.apache.catalina.valves.AccessLogValve.invoke(AccessLogValve.java:929)
        at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:118)
        at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:407)
        at org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:1002)
        at org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:585)
        at org.apache.tomcat.util.net.JIoEndpoint$SocketProcessor.run(JIoEndpoint.java:310)
        at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:885)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:907)
        at java.lang.Thread.run(Thread.java:619)
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:285)
        at sun.security.validator.PKIXValidator.engineValidate(PKIXValidator.java:191)
        at sun.security.validator.Validator.validate(Validator.java:218)
        at com.sun.net.ssl.internal.ssl.X509TrustManagerImpl.validate(X509TrustManagerImpl.java:126)
        at com.sun.net.ssl.internal.ssl.X509TrustManagerImpl.checkServerTrusted(X509TrustManagerImpl.java:209)
        at com.sun.net.ssl.internal.ssl.X509TrustManagerImpl.checkServerTrusted(X509TrustManagerImpl.java:249)
        at com.sun.net.ssl.internal.ssl.ClientHandshaker.serverCertificate(ClientHandshaker.java:954)
        ... 31 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:174)
        at java.security.cert.CertPathBuilder.build(CertPathBuilder.java:238)
        at sun.security.validator.PKIXValidator.doBuild(PKIXValidator.java:280)
        ... 37 more 

当web服务器或您连接的URL没有来自授权CA的有效证书时,会发生这种非常常见的问题。

导入服务器证书并将其安装在JDK的密钥库中,以解决此问题。

如果您打开没有有效证书的url,您可能会看到一个对话框,警告您有关证书的信息。现在单击"查看证书"并安装证书。

或者,您可以手动将证书安装到浏览器中。

在这两种情况下,我假设您已将证书导入浏览器。

现在您需要将其安装到服务器的JDK密钥库中。

通常,您将使用密钥工具来管理证书。Keytool是一个命令行实用程序,具有许多参数,允许您创建和管理用于容纳数字证书的密钥库。

转到命令行中的JDK jre/bin文件夹。在我的机器中,它的C:\Program Files\Java\jdk1.7.0_15\jre\bin

C: \Program Files\Java\jdk1.7.0_15\jre\bin>密钥工具-列表-密钥库。。\lib\security\cacerts

输入密钥库密码:changeit

上面的命令将列出所有可用的证书。

现在,您必须将上次安装的证书添加到此密钥库中。此外,首先将CA根证书导出为DER编码的二进制文件,并将其保存为D:\Root.cer在你的D:驱动器下。

现在,您可以使用以下命令将文件导入到您的cacerts密钥库中。

C: \Program Files\Java\jdk1.7.0_15\jre\bin>密钥工具-导入-别名mycertificate-密钥库。。\lib\security\cacerts-文件D:\root.cer

输入密钥库密码:changeit

信任此证书?[否]:是证书已添加到密钥库

您可以通过以下命令检查已安装的证书C: \Program Files\Java\jdk1.7.0_15\jre\bin>密钥工具-列表-密钥库。。\lib\security\cacerts

如果您找到了证书,则确认您已成功将证书安装到JDK密钥库中。

好的,我的托管提供商建议我,PayPal的一个根证书可能更新为Java密钥库中没有的证书。我应该从我的旧Java 6更新到Java 7,重新启动Tomcat以使用Java 7修复了这个问题:)

相关内容

最新更新