| 场景:今天有一个功能,spring cloud项目需要调用远程https 接口,使用 RestTemplate,本地测试无异常;部署到测试环境后报证书问题:异常如下: I/O error on POST request for "https://gateway.xxx.com/yao/user/login": sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target; nested exception is 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 
 解决办法1:因为是https证书问题,所以配置了信任所有证书,响应正常。解决办法2:代码方式证书配置代码如下:? ? @Bean(name="restTemplateRemote")
 ? ? public RestTemplate restTemplateRemote() throws Exception {
 ? ? ? ? TrustManager[] trustAllCerts =new TrustManager[] {
 ? ? ? ? ? ? ? ? new X509TrustManager() {
 ? ? ? ? ? ? ? ? ? ? @Override
 ? ? ? ? ? ? ? ? ? ? public java.security.cert.X509Certificate[] getAcceptedIssuers() {
 ? ? ? ? ? ? ? ? ? ? ? ? return new java.security.cert.X509Certificate[0];
 ? ? ? ? ? ? ? ? ? ? }
 ? ? ? ? ? ? ? ? ? ? @Override
 ? ? ? ? ? ? ? ? ? ? public void checkClientTrusted(
 ? ? ? ? ? ? ? ? ? ? ? ? ? ? java.security.cert.X509Certificate[] certs, String authType) {
 ? ? ? ? ? ? ? ? ? ? }
 ? ? ? ? ? ? ? ? ? ? @Override
 ? ? ? ? ? ? ? ? ? ? public void checkServerTrusted(
 ? ? ? ? ? ? ? ? ? ? ? ? ? ? java.security.cert.X509Certificate[] certs, String authType) {
 ? ? ? ? ? ? ? ? ? ? }
 ? ? ? ? ? ? ? ? }
 ? ? ? ? };
 ? ? ? ? SSLContext sslContext =SSLContext.getInstance("SSL");
 ? ? ? ? sslContext.init(null, trustAllCerts, new java.security.SecureRandom());
 ? ? ? ? CloseableHttpClient httpClient =HttpClients.custom()
 ? ? ? ? ? ? ? ? .setSSLContext(sslContext)
 ? ? ? ? ? ? ? ? .setSSLHostnameVerifier(NoopHostnameVerifier.INSTANCE)
 ? ? ? ? ? ? ? ? .build();
 ? ? ? ? HttpComponentsClientHttpRequestFactory customRequestFactory =new HttpComponentsClientHttpRequestFactory();
 ? ? ? ? customRequestFactory.setHttpClient(httpClient);
 ? ? ? ? return new RestTemplate(customRequestFactory);
 ? ? }
 ?
 service层代码: ??此处因为是调用第三方https接口,并不是nacos注册中心的服务,所以不用@LoadBalanced注解 @Autowired? ? @Qualifier(value = "restTemplateRemote")
 ? ? private RestTemplate restTemplate;
 接口入参出参均为json格式数据 private String refreshToken(LoginVo vo){? ? ? ? try {
 ? ? ? ? ? ? HttpHeaders httpHeaders = new HttpHeaders();
 ? ? ? ? ? ? httpHeaders.setContentType(MediaType.parseMediaType("application/json; charset=UTF-8"));
 ? ? ? ? ? ? httpHeaders.add("Accept", MediaType.APPLICATION_JSON.toString());
 ? ? ? ? ? ? HttpEntity<String> formEntity = new HttpEntity<String>(JSONObject.toJSONString(vo), httpHeaders);
 ? ? ? ? ? ? String post = restTemplate.postForObject("https://gateway.xxx.com/yao/user/login", formEntity, String.class);
 ? ? ? ? ? ? if (StringUtils.isNotBlank(post)) {
 ? ? ? ? ? ? ? ? JSONObject jsonObject = JSONObject.parseObject(post);
 ? ? ? ? ? ? ? ? if (jsonObject.getString("code").endsWith("200")) {
 ? ? ? ? ? ? ? ? ? ? JSONObject data = jsonObject.getJSONObject("data");
 ? ? ? ? ? ? ? ? ? ? String token = data.getString("token");
 ? ? ? ? ? ? ? ? ? ? return token;
 ? ? ? ? ? ? ? ? }
 ? ? ? ? ? ? }
 ? ? ? ? } catch (Exception e) {
 ? ? ? ? ? ? log.error("获取token异常:{}", e);
 ? ? ? ? ? ? throw new FundException("获取token异常:" + e.getMessage());
 ? ? ? ? }
 ? ? ? ? return null;
 ? ? }
 |