Spring Boot文档阅读笔记-how-to-implement-2-way-ssl-using-spring-boot

深碍√TFBOYSˉ_ 2022-09-02 11:49 122阅读 0赞

two-way-ssl需要12次握手(除去TCP的三次握手),如下图:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxNzg0NDI3NjE_size_16_color_FFFFFF_t_70

双向认证过程:

1.客户端发送ClientHello消息,告诉服务端要使用SSL。

2.客户端发送ServerHello的响应,告诉客户端使用SSL。

3.服务端发送证书给客户端;

4.服务端发送发送数据,要求客户端发送证书;

5.此时客户端与服务端招呼已经结束;

6.客户端发送自己的证书给服务端(客户端认证服务端证书成功);

7.客户端发送session key(使用服务端公钥加密);

8.客户端发送一个CertificateVerify消息让服务器知道它拥有发送证书;

9.客户端发送自己支持的通道加密套给服务端;

10.客户端要服务端从中选着一个;

11.服务端选择一个加密套,并且告诉客户端,我选择了这个;

12.服务端发送认证完成。

使用keytool生成证书:

  1. keytool -genkeypair -alias client-app -keyalg RSA -keysize 2048 -storetype JKS -keystore client-app.jks -validity 3650 -ext SAN=dns:localhost,ip:127.0.0.1
  2. keytool -genkeypair -alias server-app -keyalg RSA -keysize 2048 -storetype JKS -keystore server-app.jks -validity 3650 -ext SAN=dns:localhost,ip:127.0.0.1
  3. keytool -export -alias client-app -file client-app.crt -keystore client-app.jks
  4. keytool -export -alias server-app -file server-app.crt -keystore server-app.jks
  5. keytool -import -alias server-app -file server-app.crt -keystore client-app.jks
  6. keytool -import -alias client-app -file client-app.crt -keystore server-app.jks

下面使用Spring Boot进行SSL双向认证

程序结构如下:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxNzg0NDI3NjE_size_16_color_FFFFFF_t_70 1

关键代码:

服务端

application.properties

  1. spring.application.name=server-app
  2. server.port=9002
  3. server.ssl.enabled=true
  4. server.ssl.client-auth=need
  5. server.ssl.key-store=classpath:server-app.jks
  6. server.ssl.key-store-password=server-app
  7. server.ssl.key-alias=server-app
  8. server.ssl.key-store-type=JKS
  9. server.ssl.key-store-provider=SUN
  10. server.ssl.trust-store=classpath:server-app.jks
  11. server.ssl.trust-store-password=server-app
  12. server.ssl.trust-store-type=JKS

Controller.java

  1. package cn.server;
  2. import org.springframework.web.bind.annotation.GetMapping;
  3. import org.springframework.web.bind.annotation.RequestMapping;
  4. import org.springframework.web.bind.annotation.RestController;
  5. @RestController
  6. @RequestMapping("/server-app")
  7. public class Controller {
  8. @GetMapping("/data")
  9. public String getData(){
  10. System.out.println("Returning data from server-app data method");
  11. return "Hello from Server-App-data method";
  12. }
  13. }

客户端

application.properties

  1. spring.application.name=client-app
  2. server.port=9001
  3. server.ssl.enabled=true
  4. server.ssl.client-auth=need
  5. server.ssl.key-store=classpath:client-app.jks
  6. server.ssl.key-store-password=client-app
  7. server.ssl.key-alias=client-app
  8. server.ssl.key-store-type=JKS
  9. server.ssl.key-store-provider=SUN
  10. server.ssl.trust-store=classpath:client-app.jks
  11. server.ssl.trust-store-password=client-app
  12. server.ssl.trust-store-type=JKS
  13. endpoint.server-app=https://localhost:9002/server-app/data

ClientMain.java

  1. package cn.client;
  2. import org.apache.http.client.HttpClient;
  3. import org.apache.http.conn.ssl.NoopHostnameVerifier;
  4. import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
  5. import org.apache.http.conn.ssl.TrustSelfSignedStrategy;
  6. import org.apache.http.impl.client.HttpClients;
  7. import org.apache.http.ssl.SSLContextBuilder;
  8. import org.springframework.beans.factory.annotation.Autowired;
  9. import org.springframework.beans.factory.annotation.Value;
  10. import org.springframework.boot.CommandLineRunner;
  11. import org.springframework.boot.SpringApplication;
  12. import org.springframework.boot.autoconfigure.SpringBootApplication;
  13. import org.springframework.context.annotation.Bean;
  14. import org.springframework.core.io.ClassPathResource;
  15. import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
  16. import org.springframework.web.client.RestTemplate;
  17. import java.io.InputStream;
  18. import java.net.URI;
  19. import java.security.KeyStore;
  20. @SpringBootApplication
  21. public class ClientMain implements CommandLineRunner {
  22. @Autowired
  23. RestTemplate restTemplate;
  24. @Value("${endpoint.server-app}")
  25. private String msEndpoint;
  26. @Bean
  27. public RestTemplate getRestTemplate(){
  28. RestTemplate restTemplate = new RestTemplate();
  29. KeyStore keyStore;
  30. HttpComponentsClientHttpRequestFactory requestFactory = null;
  31. try {
  32. keyStore = KeyStore.getInstance("jks");
  33. ClassPathResource classPathResource = new ClassPathResource("client-app.jks");
  34. InputStream inputStream = classPathResource.getInputStream();
  35. keyStore.load(inputStream, "client-app".toCharArray());
  36. SSLConnectionSocketFactory socketFactory = new SSLConnectionSocketFactory(new SSLContextBuilder()
  37. .loadTrustMaterial(null, new TrustSelfSignedStrategy())
  38. .loadKeyMaterial(keyStore, "client-app".toCharArray()).build(),
  39. NoopHostnameVerifier.INSTANCE);
  40. HttpClient httpClient = HttpClients.custom().setSSLSocketFactory(socketFactory)
  41. .setMaxConnTotal(Integer.valueOf(5))
  42. .setMaxConnPerRoute(Integer.valueOf(5))
  43. .build();
  44. requestFactory = new HttpComponentsClientHttpRequestFactory(httpClient);
  45. requestFactory.setReadTimeout(Integer.valueOf(10000));
  46. requestFactory.setConnectTimeout(Integer.valueOf(10000));
  47. restTemplate.setRequestFactory(requestFactory);
  48. }
  49. catch (Exception exception) {
  50. System.out.println("Exception Occured while creating restTemplate "+exception);
  51. exception.printStackTrace();
  52. }
  53. return restTemplate;
  54. }
  55. public static void main(String[] args) {
  56. SpringApplication.run(ClientMain.class, args);
  57. }
  58. @Override
  59. public void run(String... args) throws Exception {
  60. String forObject = restTemplate.getForObject(new URI(msEndpoint), String.class);
  61. System.out.println(forObject);
  62. }
  63. }

源码打包下载地址:

https://github.com/fengfanchen/Java/tree/master/SpringBoot-two-way-ssl

发表评论

表情:
评论列表 (有 0 条评论,122人围观)

还没有评论,来说两句吧...

相关阅读