For a JAX-WS client to accept all SSL certificates from any server, you must create an SLLSocketFactory that contains an X509TrustManager that bypasses all certificate validation.
Building the X509TrustManager is simple. Just create a class that implements the X509TrustManager interface and essentially do nothing in the methods:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import java.security.cert.CertificateException; | |
import java.security.cert.X509Certificate; | |
import javax.net.ssl.X509TrustManager; | |
public class BlindTrustManager implements X509TrustManager { | |
public X509Certificate[] getAcceptedIssuers() { | |
return null; | |
} | |
public void checkClientTrusted(X509Certificate[] chain, String authType) | |
throws CertificateException { | |
} | |
public void checkServerTrusted(X509Certificate[] chain, String authType) | |
throws CertificateException { | |
} | |
} |
If Host Name validation is desired, this can be implementd in the checkServerTrusted method. You can obtain the server Host Name from the last certificate in the passed-in array of X509Certificates.
Once your TrustManager is set up, the next step is to create an SSLSocketFactory. The factory is created by getting an instance of SSLContext and initializing it with your X509TrustManager you created previously. Then calling getSocketFactory on the context will return your SSLSocketFactory.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private SSLSocketFactory buildSocketFactory() throws KeyManagementException, NoSuchAlgorithmException { | |
SSLContext ctx = SSLContext.getInstance("TLS"); | |
ctx.init(null, new TrustManager[] { new BlindTrustManager() }, null); | |
return ctx.getSocketFactory(); | |
} |
I recommended caching the SSLSocketFactory instance rather than creating a new one for each request because initialization can be time-consuming.
After the SSLSocketFactory is constructed, all that is left is to add it to your requestContext of your service proxy.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
SSLSocketFactory sslSocketFactory = buildSocketFactory(); | |
MyJAXService service = new MyJAXService(); | |
MyServicePortProxy serviceProxy = service.getMyServicePortProxy(); | |
Map<String, Object> requestContext = ((BindingProvider) serviceProxy).getRequestContext(); | |
requestContext.put(BindingProviderProperties.SSL_SOCKET_FACTORY, sslSocketFactory); |
Your client should now be able to communicate with servers over SSL without needing to specify a trust store or add any new certificates to your existing trust store