HttpGet с HTTPS: SSLPeerUnverifiedException
Используя HttpClient, я получаю следующую ошибку при попытке связи через HTTPS:
Исключение в потоке "main" javax.net.ssl.SSLPeerUnverifiedException: peer не аутентифицирован.
Вот мой код:
URI loginUri = new URI("https://myUrl.asp");
HttpClient httpclient = new DefaultHttpClient();
HttpGet httpget = new HttpGet( loginUri );
HttpResponse response = httpclient.execute( httpget );
Как подавить или удалить эту ошибку?
Ответы
Ответ 1
Используя HttpClient 3.x, вам нужно сделать это:
Protocol easyHttps = new Protocol("https", new EasySSLProtocolSocketFactory(), 443);
Protocol.registerProtocol("https", easyHttps);
Реализация EasySSLProtocolSocketFactory можно найти здесь.
Ответ 2
Примечание. Не делайте этого в производственном коде, используйте вместо него http или сам открытый открытый ключ, как было предложено выше.
В HttpClient 4.xx:
import static org.junit.Assert.assertEquals;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.X509Certificate;
import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.conn.scheme.Scheme;
import org.apache.http.conn.ssl.SSLSocketFactory;
import org.apache.http.impl.client.DefaultHttpClient;
import org.junit.Test;
public class HttpClientTrustingAllCertsTest {
@Test
public void shouldAcceptUnsafeCerts() throws Exception {
DefaultHttpClient httpclient = httpClientTrustingAllSSLCerts();
HttpGet httpGet = new HttpGet("https://host_with_self_signed_cert");
HttpResponse response = httpclient.execute( httpGet );
assertEquals("HTTP/1.1 200 OK", response.getStatusLine().toString());
}
private DefaultHttpClient httpClientTrustingAllSSLCerts() throws NoSuchAlgorithmException, KeyManagementException {
DefaultHttpClient httpclient = new DefaultHttpClient();
SSLContext sc = SSLContext.getInstance("SSL");
sc.init(null, getTrustingManager(), new java.security.SecureRandom());
SSLSocketFactory socketFactory = new SSLSocketFactory(sc);
Scheme sch = new Scheme("https", 443, socketFactory);
httpclient.getConnectionManager().getSchemeRegistry().register(sch);
return httpclient;
}
private TrustManager[] getTrustingManager() {
TrustManager[] trustAllCerts = new TrustManager[] { new X509TrustManager() {
@Override
public java.security.cert.X509Certificate[] getAcceptedIssuers() {
return null;
}
@Override
public void checkClientTrusted(X509Certificate[] certs, String authType) {
// Do nothing
}
@Override
public void checkServerTrusted(X509Certificate[] certs, String authType) {
// Do nothing
}
} };
return trustAllCerts;
}
}
Ответ 3
Этот ответ следует owlstead и Mat. Это относится к установкам SE/EE, а не к ME/mobile/Android SSL.
Поскольку никто еще не упомянул об этом, я упомянул "производственный путь", чтобы исправить это:
Выполните шаги из класса AuthSSLProtocolSocketFactory в HttpClient, чтобы обновить хранилище хранилищ и хранилища ключей.
- Импортировать доверенный сертификат и создать файл доверия.
keytool -import -alias "my server cert" -file server.crt -keystore my.truststore
- Создайте новый ключ (используйте тот же пароль, что и доверительный магазин)
keytool -genkey -v -alias "my client key" -validity 365 -keystore my.keystore
- Выдать запрос на подпись сертификата (CSR)
keytool -certreq -alias "my client key" -file mycertreq.csr -keystore my.keystore
-
(самозапись или подписка на сертификат)
-
Импорт доверенного корневого сертификата CA
keytool -import -alias "my trusted ca" -file caroot.crt -keystore my.keystore
- Импортировать файл PKCS # 7 в общую цепочку сертификатов
keytool -import -alias "my client key" -file mycert.p7 -keystore my.keystore
- Проверить итоговое содержимое файла хранилища ключей
keytool -list -v -keystore my.keystore
Если у вас нет сертификата сервера, сгенерируйте его в формате JKS, а затем экспортируйте его как файл CRT. Источник: документация по keytool
keytool -genkey -alias server-alias -keyalg RSA -keypass changeit
-storepass changeit -keystore my.keystore
keytool -export -alias server-alias -storepass changeit
-file server.crt -keystore my.keystore
Ответ 4
Это исключение произойдет, если ваш сервер основан на JDK 7, а ваш клиент находится на JDK 6 и использует SSL-сертификаты. В JDK 7 сообщение sslv2hello handshaking отключено по умолчанию, в то время как в JDK 6 sslv2hello сообщение об отказе включено. По этой причине, когда ваш клиент пытается подключиться к серверу, сообщение sslv2hello будет отправлено на сервер, и из-за отключения сообщения sslv2hello вы получите это исключение. Чтобы решить эту проблему, вы должны переместить своего клиента в JDK 7 или вам нужно использовать версию JuK версии 6u91. Но чтобы получить эту версию JDK, вы должны получить
Ответ 5
Метод возвращает "secureClient" (в среде Java 7 - NetBeans IDE и GlassFish Server: порт https по умолчанию 3920), надеюсь, что это может помочь:
public DefaultHttpClient secureClient() {
DefaultHttpClient httpclient = new DefaultHttpClient();
SSLSocketFactory sf;
KeyStore trustStore;
FileInputStream trustStream = null;
File truststoreFile;
// java.security.cert.PKIXParameters for the trustStore
PKIXParameters pkixParamsTrust;
KeyStore keyStore;
FileInputStream keyStream = null;
File keystoreFile;
// java.security.cert.PKIXParameters for the keyStore
PKIXParameters pkixParamsKey;
try {
trustStore = KeyStore.getInstance(KeyStore.getDefaultType());
truststoreFile = new File(TRUSTSTORE_FILE);
keyStore = KeyStore.getInstance(KeyStore.getDefaultType());
keystoreFile = new File(KEYSTORE_FILE);
try {
trustStream = new FileInputStream(truststoreFile);
keyStream = new FileInputStream(keystoreFile);
} catch (FileNotFoundException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
}
try {
trustStore.load(trustStream, PASSWORD.toCharArray());
keyStore.load(keyStream, PASSWORD.toCharArray());
} catch (IOException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
} catch (CertificateException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
}
try {
pkixParamsTrust = new PKIXParameters(trustStore);
// accepts Server certificate generated with keytool and (auto) signed by SUN
pkixParamsTrust.setPolicyQualifiersRejected(false);
} catch (InvalidAlgorithmParameterException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
}
try {
pkixParamsKey = new PKIXParameters(keyStore);
// accepts Client certificate generated with keytool and (auto) signed by SUN
pkixParamsKey.setPolicyQualifiersRejected(false);
} catch (InvalidAlgorithmParameterException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
}
try {
sf = new SSLSocketFactory(trustStore);
ClientConnectionManager manager = httpclient.getConnectionManager();
manager.getSchemeRegistry().register(new Scheme("https", 3920, sf));
} catch (KeyManagementException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
} catch (UnrecoverableKeyException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
}
} catch (NoSuchAlgorithmException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
} catch (KeyStoreException ex) {
Logger.getLogger(ApacheHttpRestClient.class.getName()).log(Level.SEVERE, null, ex);
}
// use the httpclient for any httpRequest
return httpclient;
}