在Java中实现双向认证,通常涉及到两个实体之间的相互认证。一个常见的场景是客户端和服务器之间的安全通信。以下是一个简单的示例,展示了如何使用Java实现双向认证。
1. 服务器端
首先,我们需要创建一个服务器,该服务器将使用SSL/TLS来保护通信,并使用自定义的X509TrustManager
来验证客户端证书。
import javax.net.ServerSocketFactory; import javax.net.ssl.*; import java.io.FileInputStream; import java.security.KeyStore; public class SecureServer { public static void main(String[] args) throws Exception { // 加载服务器证书 KeyStore keyStore = KeyStore.getInstance("JKS"); keyStore.load(new FileInputStream("path/to/server.jks"), "password".toCharArray()); // 创建KeyManagerFactory KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm()); keyManagerFactory.init(keyStore, "password".toCharArray()); // 创建TrustManagerFactory TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm()); trustManagerFactory.init(keyStore); // 创建SSLContext SSLContext sslContext = SSLContext.getInstance("TLS"); sslContext.init(keyManagerFactory.getKeyManagers(), trustManagerFactory.getTrustManagers(), null); // 创建ServerSocketFactory ServerSocketFactory serverSocketFactory = sslContext.getServerSocketFactory(); // 启动服务器 try (ServerSocket serverSocket = serverSocketFactory.createServerSocket(8443)) { System.out.println("Server started on port 8443"); while (true) { try (Socket socket = serverSocket.accept()) { System.out.println("Client connected"); // 处理客户端请求 } catch (Exception e) { e.printStackTrace(); } } } } }
2. 客户端
接下来,我们创建一个客户端,该客户端将使用SSL/TLS来保护通信,并使用自定义的X509TrustManager
来验证服务器证书。
import javax.net.ssl.*; import java.io.FileInputStream; import java.security.KeyStore; import java.security.cert.CertificateException; import java.security.cert.X509Certificate; public class SecureClient { public static void main(String[] args) throws Exception { // 加载客户端证书 KeyStore keyStore = KeyStore.getInstance("JKS"); keyStore.load(new FileInputStream("path/to/client.jks"), "password".toCharArray()); // 创建KeyManagerFactory KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm()); keyManagerFactory.init(keyStore, "password".toCharArray()); // 创建TrustManager TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm()); trustManagerFactory.init(keyStore); // 创建SSLContext SSLContext sslContext = SSLContext.getInstance("TLS"); sslContext.init(keyManagerFactory.getKeyManagers(), trustManagerFactory.getTrustManagers(), null); // 创建SSLSocketFactory SSLSocketFactory socketFactory = sslContext.getSocketFactory(); // 连接到服务器 try (SSLSocket socket = (SSLSocket) socketFactory.createSocket("localhost", 8443)) { System.out.println("Connected to server"); // 发送和接收数据 // ... } catch (Exception e) { e.printStackTrace(); } } }
3. 自定义TrustManager
为了实现双向认证,我们需要自定义X509TrustManager
来验证客户端和服务器的证书。
import javax.net.ssl.*; import java.security.cert.CertificateException; import java.security.cert.X509Certificate; public class CustomX509TrustManager implements X509TrustManager { @Override public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException { // 验证客户端证书的逻辑 } @Override public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException { // 验证服务器证书的逻辑 } @Override public X509Certificate[] getAcceptedIssuers() { return new X509Certificate[0]; } }
4. 配置TrustManager
在服务器和客户端中,我们需要将自定义的X509TrustManager
配置到SSLContext
中。
// 服务器端 TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm()); trustManagerFactory.init(keyStore); trustManagerFactory.setTrustManagers(new CustomX509TrustManager()); // 客户端 TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm()); trustManagerFactory.init(keyStore); trustManagerFactory.setTrustManagers(new CustomX509TrustManager());
通过以上步骤,我们实现了一个简单的双向认证示例。在实际应用中,你可能需要根据具体需求进行更多的配置和错误处理。