Android

Https Connection Android

Https-соединение Android

Я публикую сообщение https и получаю исключение из ssl-исключения, не являющегося доверенным сертификатом сервера. Если я использую обычный http, он работает отлично. Должен ли я каким-либо образом принимать сертификат сервера?

Переведено автоматически
Ответ 1

Я скопировал следующий код, который взят из The Java Developers Almanac.

Он просто больше не проверяет сертификат.


// always verify the host - dont check for certificate final static
HostnameVerifier DO_NOT_VERIFY = new HostnameVerifier() { public
boolean verify(String hostname, SSLSession session) { return true;
} };

/** * Trust every server - dont check for any certificate */ private
static void trustAllHosts() { // Create a trust manager that does not
validate certificate chains TrustManager[] trustAllCerts = new
TrustManager[] { new X509TrustManager() { public
java.security.cert.X509Certificate[] getAcceptedIssuers() { return
new java.security.cert.X509Certificate[] {}; }

public void checkClientTrusted(X509Certificate[] chain,
String authType)
throws CertificateException { }

public void checkServerTrusted(X509Certificate[] chain,
String authType)
throws CertificateException { } } };

// Install the all-trusting trust manager try { SSLContext sc =
SSLContext.getInstance("TLS"); sc.init(null, trustAllCerts, new
java.security.SecureRandom()); HttpsURLConnection
.setDefaultSSLSocketFactory(sc.getSocketFactory()); } catch (Exception e) { e.printStackTrace(); } }

и


HttpURLConnection http = null;

if (url.getProtocol().toLowerCase().equals("https")) {
trustAllHosts(); HttpsURLConnection https = (HttpsURLConnection) url.openConnection(); https.setHostnameVerifier(DO_NOT_VERIFY); http
= https; } else { http = (HttpURLConnection) url.openConnection(); }

Ответ 2

Я делаю предположение, но если вы хотите, чтобы произошло реальное рукопожатие, вы должны сообщить Android о своем сертификате. Если вы хотите просто принять, несмотря ни на что, то используйте этот псевдокод, чтобы получить то, что вам нужно, с помощью HTTP-клиента Apache:

SchemeRegistry schemeRegistry = new SchemeRegistry ();

schemeRegistry.register (new Scheme ("http",
PlainSocketFactory.getSocketFactory (), 80));
schemeRegistry.register (new Scheme ("https",
new CustomSSLSocketFactory (), 443));

ThreadSafeClientConnManager cm = new ThreadSafeClientConnManager (
params, schemeRegistry);


return new DefaultHttpClient (cm, params);

CustomSSLSocketFactory:

public class CustomSSLSocketFactory extends org.apache.http.conn.ssl.SSLSocketFactory
{
private SSLSocketFactory FACTORY = HttpsURLConnection.getDefaultSSLSocketFactory ();

public CustomSSLSocketFactory ()
{
super(null);
try
{
SSLContext context = SSLContext.getInstance ("TLS");
TrustManager[] tm = new TrustManager[] { new FullX509TrustManager () };
context.init (null, tm, new SecureRandom ());

FACTORY = context.getSocketFactory ();
}
catch (Exception e)
{
e.printStackTrace();
}
}

public Socket createSocket() throws IOException
{
return FACTORY.createSocket();
}

// TODO: add other methods like createSocket() and getDefaultCipherSuites().
// Hint: they all just make a call to member FACTORY
}

FullX509TrustManager - это класс, который реализует javax.net.ssl.X509TrustManager, но ни один из методов фактически не выполняет никакой работы, получите образец здесь.

Удачи!

Ответ 3

http://madurangasblogs.blogspot.in/2013/08/avoiding-javaxnetsslsslpeerunverifiedex.html

Предоставлено Мадурангой

При разработке приложения, использующего https, на вашем тестовом сервере нет действительного SSL-сертификата. Иногда веб-сайт использует самозаверяющий сертификат или веб-сайт использует бесплатный SSL-сертификат. Итак, если вы попытаетесь подключиться к серверу с помощью Apache HttpClient, вы получите исключение, сообщающее, что "одноранговый узел не прошел проверку подлинности". Хотя не рекомендуется доверять всем сертификатам в производственном программном обеспечении, возможно, вам придется делать это в зависимости от ситуации. Это решение устраняет исключение, вызванное "одноранговый узел не прошел проверку подлинности".

But before we go to the solution, I must warn you that this is not a good idea for a production application. This will violate the purpose of using a security certificate. So unless you have a good reason or if you are sure that this will not cause any problem, don't use this solution.

Normally you create a HttpClient like this.

HttpClient httpclient = new DefaultHttpClient();

But you have to change the way you create the HttpClient.

First you have to create a class extending org.apache.http.conn.ssl.SSLSocketFactory.

import org.apache.http.conn.ssl.SSLSocketFactory;
import java.io.IOException;
import java.net.Socket;
import java.net.UnknownHostException;
import java.security.KeyManagementException;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.UnrecoverableKeyException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;

import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;

public class MySSLSocketFactory extends SSLSocketFactory {
SSLContext sslContext = SSLContext.getInstance("TLS");

public MySSLSocketFactory(KeyStore truststore) throws NoSuchAlgorithmException, KeyManagementException, KeyStoreException, UnrecoverableKeyException {
super(truststore);

TrustManager tm = new X509TrustManager() {
public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
}

public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
}

public X509Certificate[] getAcceptedIssuers() {
return null;
}
};

sslContext.init(null, new TrustManager[] { tm }, null);
}

@Override
public Socket createSocket(Socket socket, String host, int port, boolean autoClose) throws IOException, UnknownHostException {
return sslContext.getSocketFactory().createSocket(socket, host, port, autoClose);
}

@Override
public Socket createSocket() throws IOException {
return sslContext.getSocketFactory().createSocket();
}
}

Затем создайте метод, подобный этому.

public HttpClient getNewHttpClient() {
try {
KeyStore trustStore = KeyStore.getInstance(KeyStore.getDefaultType());
trustStore.load(null, null);

SSLSocketFactory sf = new MySSLSocketFactory(trustStore);
sf.setHostnameVerifier(SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER);

HttpParams params = new BasicHttpParams();
HttpProtocolParams.setVersion(params, HttpVersion.HTTP_1_1);
HttpProtocolParams.setContentCharset(params, HTTP.UTF_8);

SchemeRegistry registry = new SchemeRegistry();
registry.register(new Scheme("http", PlainSocketFactory.getSocketFactory(), 80));
registry.register(new Scheme("https", sf, 443));

ClientConnectionManager ccm = new ThreadSafeClientConnManager(params, registry);

return new DefaultHttpClient(ccm, params);
} catch (Exception e) {
return new DefaultHttpClient();
}
}

Затем вы можете создать HttpClient.

HttpClient httpclient = getNewHttpClient();

Если вы пытаетесь отправить post-запрос на страницу входа в систему, остальная часть кода будет выглядеть следующим образом.

private URI url = new URI("url of the action of the form");
HttpPost httppost = new HttpPost(url);
List<NameValuePair> nameValuePairs = new ArrayList<NameValuePair>();
nameValuePairs.add(new BasicNameValuePair("username", "user"));
nameValuePairs.add(new BasicNameValuePair("password", "password"));
try {
httppost.setEntity(new UrlEncodedFormEntity(nameValuePairs));
HttpResponse response = httpclient.execute(httppost);
HttpEntity entity = response.getEntity();
InputStream is = entity.getContent();
} catch (UnsupportedEncodingException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (ClientProtocolException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

Вы получаете HTML-страницу в InputStream. Затем вы можете делать все, что хотите, с возвращенной HTML-страницей.

Но здесь вы столкнетесь с проблемой. Если вы хотите управлять сеансом с помощью файлов cookie, вы не сможете сделать это этим методом. Если вы хотите получить файлы cookie, вам придется сделать это через браузер. Тогда только вы будете получать файлы cookie.

Ответ 4

Если вы используете сертификат StartSSL или Thawte, для Froyo и более старых версий произойдет сбой. Вы можете использовать репозиторий CAcert более новой версии вместо того, чтобы доверять каждому сертификату.

2023-07-14 00:12 java android ssl