問題描述
使用RestTemplate發送HTTPS請求的時候,出現了這樣的一個問題:
sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target
RestTemplate 默認不支持https協議
解決方案:
????????第一種是忽略認證
????????第二種是導入證書,比較復雜(比第一種安全)?
解決方案:
這里說一下第一種解決方案,忽略認證
RestTemplateConfig
package com.test.config;import java.nio.charset.Charset;
import java.util.List;import javax.net.ssl.SSLContext;import org.apache.http.conn.ssl.NoopHostnameVerifier;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.ssl.SSLContexts;
import org.apache.http.ssl.TrustStrategy;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.StringHttpMessageConverter;
import org.springframework.web.client.RestTemplate;@Configuration
public class RestTemplateConfig{@Bean("restTemplate")public RestTemplate RestTemplate() {HttpComponentsClientHttpRequestFactory httpRequestFactory = new HttpComponentsClientHttpRequestFactory();httpRequestFactory.setConnectionRequestTimeout(30000);httpRequestFactory.setConnectTimeout(30000);httpRequestFactory.setReadTimeout(30000);return new RestTemplate(httpRequestFactory);}/*** 用于https請求,忽略認證* @return unSSLRestTemplate*/@Bean("unSSLRestTemplate")public RestTemplate restTemplateHttps() {RestTemplate restTemplate = null;try {TrustStrategy acceptingTrustStrategy = (chain, authType) -> true;SSLContext sslContext = SSLContexts.custom().loadTrustMaterial(null, acceptingTrustStrategy).build();SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslContext, NoopHostnameVerifier.INSTANCE);HttpClientBuilder clientBuilder = HttpClients.custom();CloseableHttpClient httpClient = clientBuilder.setSSLSocketFactory(sslsf).build();HttpComponentsClientHttpRequestFactory httpRequestFactory = new HttpComponentsClientHttpRequestFactory();httpRequestFactory.setConnectionRequestTimeout(30000);httpRequestFactory.setConnectTimeout(30000);httpRequestFactory.setReadTimeout(30000);httpRequestFactory.setHttpClient(httpClient);restTemplate = new RestTemplate(httpRequestFactory);//解決亂碼List<HttpMessageConverter<?>> httpMessageConverters = restTemplate.getMessageConverters();httpMessageConverters.stream().forEach(httpMessageConverter ->{if(httpMessageConverter instanceof StringHttpMessageConverter){StringHttpMessageConverter messageConverter = (StringHttpMessageConverter)httpMessageConverter;messageConverter.setDefaultCharset(Charset.forName("UTF-8"));}});} catch (Exception e) {e.printStackTrace();}return restTemplate;}
}
測試代碼
package com.test.service;import javax.annotation.Resource;import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;/*** http請求&https請求*/
@Service
public class TestService {//http請求@Resource(name = "restTemplate")private RestTemplate restTemplate;//https請求@Resource(name = "unSSLRestTemplate")private RestTemplate unSSLRestTemplate;/*** http請求*/public void interfaceHttp(JSONObject params) {//參數String json = params.toJSONString();//請求頭HttpHeaders headers = new HttpHeaders();headers.add("Content-Type", MediaType.APPLICATION_JSON_VALUE);HttpEntity<String> formEntity = new HttpEntity<String>(json, headers);restTemplate.postForObject(URL, formEntity, String.class);}/*** https請求*/public void interfaceHttps(JSONObject params) {//參數String json = params.toJSONString();//請求頭HttpHeaders headers = new HttpHeaders();headers.add("Content-Type", MediaType.APPLICATION_JSON_VALUE);HttpEntity<String> formEntity = new HttpEntity<String>(json, headers);unSSLRestTemplate.postForObject(URL, formEntity, String.class);}
}
說明:這里兼容http和https請求,只需要指定名稱即可?
?