httpclient ssl сертификат на андроид - PullRequest
2 голосов
/ 19 мая 2010

У меня есть некоторые проблемы с ssl при использовании httpclient на Android. Я пытаюсь получить подробный доступ к самозаверяющему сертификату. Я хочу, чтобы мое приложение доверяло всем сертификатам (я буду использовать ssl только для шифрования данных). Сначала я попытался использовать это руководство http://hc.apache.org/httpclient-3.x/sslguide.html на рабочем столе работает нормально, но на Android я все еще получил javax.net.ssl.SSLException: не доверенный сертификат сервера. После поиска в Google я нашел несколько других примеров, как включить ssl.

http://groups.google.com/group/android-developers/browse_thread/thread/62d856cdcfa9f16e - Работа, когда я использую URLConnection, но с HttpClient все еще получил исключение.

http://www.discursive.com/books/cjcook/reference/http-webdav-sect-self-signed.html - на рабочем столе, использующем jar из apache, работает, но в Android использование включенных в классы SDK не может заставить его работать.

http://mail-archives.apache.org/mod_mbox/hc-httpclient-users/200808.mbox/%3C1218824624.6561.14.camel@ubuntu%3E - также получите то же исключение

Так что любые идеи, как я могу доверять всем сертификатам на Android с помощью HttpClient

Ответы [ 3 ]

9 голосов
/ 19 мая 2010

Если вы посмотрите на код DefaultHttpClient, он будет выглядеть примерно так:

   @Override
    protected ClientConnectionManager createClientConnectionManager() {
        SchemeRegistry registry = new SchemeRegistry();
        registry.register(
                new Scheme("http", PlainSocketFactory.getSocketFactory(), 80));
        registry.register(
                new Scheme("https", SSLSocketFactory.getSocketFactory(), 443));

        ClientConnectionManager connManager = null;     
        HttpParams params = getParams();
        ...
    }

Обратите внимание на сопоставление схемы https с org.apache.http.conn.ssl.SSLSocketFactory.getSocketFactory ().

Вы можете создать пользовательскую реализацию для интерфейса org.apache.commons.httpclient.protocol.SecureProtocolSocketFactory (http://hc.apache.org/httpclient-3.x/apidocs/org/apache/commons/httpclient/protocol/SecureProtocolSocketFactory.html), в которой вы можете создать java.net.SSLSocket с пользовательским TrustManager, который принимает все сертификаты.

Возможно, вы захотите ознакомиться с JSSE для получения более подробной информации по адресу http://java.sun.com/j2se/1.4.2/docs/guide/security/jsse/JSSERefGuide.html

3 голосов
/ 26 апреля 2011

Ключевой идеей является использование настроенного SSLSocketFactory, реализующего LayeredSocketFactory. Для настроенного сокета не требуется HostNameVerifier.

private static final class TrustAllSSLSocketFactory implements
    LayeredSocketFactory {

    private static final TrustAllSSLSocketFactory DEFAULT_FACTORY = new TrustAllSSLSocketFactory();

    public static TrustAllSSLSocketFactory getSocketFactory() {
        return DEFAULT_FACTORY;
    }

    private SSLContext sslcontext;
    private javax.net.ssl.SSLSocketFactory socketfactory;

    private TrustAllSSLSocketFactory() {
        super();
        TrustManager[] tm = new TrustManager[] { new X509TrustManager() {

            @Override
            public void checkClientTrusted(X509Certificate[] chain,
                String authType) throws CertificateException {
                // do nothing
            }

            @Override
            public void checkServerTrusted(X509Certificate[] chain,
                String authType) throws CertificateException {
                // do nothing
            }

            @Override
            public X509Certificate[] getAcceptedIssuers() {
                return new X509Certificate[0];
            }

        } };
        try {
            this.sslcontext = SSLContext.getInstance(SSLSocketFactory.TLS);
            this.sslcontext.init(null, tm, new SecureRandom());
            this.socketfactory = this.sslcontext.getSocketFactory();
        } catch ( NoSuchAlgorithmException e ) {
            Log.e(LOG_TAG,
                "Failed to instantiate TrustAllSSLSocketFactory!", e);
        } catch ( KeyManagementException e ) {
            Log.e(LOG_TAG,
                "Failed to instantiate TrustAllSSLSocketFactory!", e);
        }
    }

    @Override
    public Socket createSocket(Socket socket, String host, int port,
        boolean autoClose) throws IOException, UnknownHostException {
        SSLSocket sslSocket = (SSLSocket) this.socketfactory.createSocket(
            socket, host, port, autoClose);
        return sslSocket;
    }

    @Override
    public Socket connectSocket(Socket sock, String host, int port,
        InetAddress localAddress, int localPort, HttpParams params)
        throws IOException, UnknownHostException, ConnectTimeoutException {
        if ( host == null ) {
            throw new IllegalArgumentException(
                "Target host may not be null.");
        }
        if ( params == null ) {
            throw new IllegalArgumentException(
                "Parameters may not be null.");
        }

        SSLSocket sslsock = (SSLSocket) ( ( sock != null ) ? sock
            : createSocket() );

        if ( ( localAddress != null ) || ( localPort > 0 ) ) {

            // we need to bind explicitly
            if ( localPort < 0 ) {
                localPort = 0; // indicates "any"
            }

            InetSocketAddress isa = new InetSocketAddress(localAddress,
                localPort);
            sslsock.bind(isa);
        }

        int connTimeout = HttpConnectionParams.getConnectionTimeout(params);
        int soTimeout = HttpConnectionParams.getSoTimeout(params);

        InetSocketAddress remoteAddress;
        remoteAddress = new InetSocketAddress(host, port);

        sslsock.connect(remoteAddress, connTimeout);

        sslsock.setSoTimeout(soTimeout);

        return sslsock;
    }

    @Override
    public Socket createSocket() throws IOException {
        // the cast makes sure that the factory is working as expected
        return (SSLSocket) this.socketfactory.createSocket();
    }

    @Override
    public boolean isSecure(Socket sock) throws IllegalArgumentException {
        return true;
    }

}

Затем вы можете продолжать использовать настроенный SSLSocketFactory в реестре поддерживаемых схем.

private static final BasicHttpParams sHttpParams = new BasicHttpParams();
private static final SchemeRegistry sSupportedSchemes = new SchemeRegistry();
static {
    sHttpParams.setParameter("http.socket.timeout", READ_TIMEOUT);
    sHttpParams.setParameter("http.connection.timeout", CONNECT_TIMEOUT);
    sSupportedSchemes.register(new Scheme("http",
        PlainSocketFactory.getSocketFactory(), 80));
    sSupportedSchemes.register(new Scheme("https",
        TrustAllSSLSocketFactory.getSocketFactory(), 443));
}
2 голосов
/ 17 июня 2011

Вместо того, чтобы принимать все сертификаты, я рекомендую это решение: Доверие всем сертификатам с использованием HttpClient поверх HTTPS

...