Problem
I want to send https request to the site https://10.2.20.20/fido/EzPay/login.php my own server and get response from it and save it for example in a string. I have found some example codes in internet and try to test them for my problem but they are not helpful. Below I present some parts of codes which I have tested.
Code Example:
I try this code, but I always get same exception “No peer certificate” Why ?
try
{
HostnameVerifier hostnameVerifier = org.apache.http.conn.ssl.SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER;
DefaultHttpClient client = new DefaultHttpClient();
SchemeRegistry registry = new SchemeRegistry();
SSLSocketFactory socketFactory = SSLSocketFactory.getSocketFactory();
socketFactory.setHostnameVerifier((X509HostnameVerifier) hostnameVerifier);
registry.register(new Scheme("https", socketFactory, 443));
SingleClientConnManager mgr = new SingleClientConnManager(client.getParams(), registry);
DefaultHttpClient httpClient = new DefaultHttpClient(mgr, client.getParams());
// Set verifier
HttpsURLConnection.setDefaultHostnameVerifier(hostnameVerifier);
// Example send http request
final String url = "https://10.2.20.20/fido/EzPay/login.php";
HttpPost httpPost = new HttpPost(url);
HttpResponse response = httpClient.execute(httpPost);
BufferedReader rd = new BufferedReader(new InputStreamReader(response.getEntity().getContent()));
String line = "";
while ((line = rd.readLine()) != null) {
Log.i(DownloadImageTask.class.getName(), line);
}
}
catch(IOException ex)
{
Log.e(DownloadImageTask.class.getName(), ex.getMessage());
}
Exception.
03-02 16:58:25.234: W/System.err(1868):
javax.net.ssl.SSLPeerUnverifiedException: No peer certificate 03-02
16:58:25.238: W/System.err(1868): at
org.apache.harmony.xnet.provider.jsse.SSLSessionImpl.getPeerCertificates(SSLSessionImpl.java:137)
03-02 16:58:25.238: W/System.err(1868): at
org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:93)
03-02 16:58:25.238: W/System.err(1868): at
org.apache.http.conn.ssl.SSLSocketFactory.createSocket(SSLSocketFactory.java:381)
03-02 16:58:25.238: W/System.err(1868): at
org.apache.http.impl.conn.DefaultClientConnectionOperator.openConnection(DefaultClientConnectionOperator.java:165)
03-02 16:58:25.250: W/System.err(1868): at
org.apache.http.impl.conn.AbstractPoolEntry.open(AbstractPoolEntry.java:164)
03-02 16:58:25.250: W/System.err(1868): at
org.apache.http.impl.conn.AbstractPooledConnAdapter.open(AbstractPooledConnAdapter.java:119)
03-02 16:58:25.250: W/System.err(1868): at
org.apache.http.impl.client.DefaultRequestDirector.execute(DefaultRequestDirector.java:360)
03-02 16:58:25.250: W/System.err(1868): at
org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:555)
03-02 16:58:25.250: W/System.err(1868): at
org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:487)
03-02 16:58:25.250: W/System.err(1868): at
org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:465)
03-02 16:58:25.250: W/System.err(1868): at
com.https.test.DownloadImageTask.doInBackground(Https_testActivity.java:78)
03-02 16:58:25.250: W/System.err(1868): at
com.https.test.DownloadImageTask.doInBackground(Https_testActivity.java:1)
03-02 16:58:25.250: W/System.err(1868): at
android.os.AsyncTask$2.call(AsyncTask.java:264) 03-02 16:58:25.253:
W/System.err(1868): at
java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
03-02 16:58:25.253: W/System.err(1868): at
java.util.concurrent.FutureTask.run(FutureTask.java:137) 03-02
16:58:25.253: W/System.err(1868): at
android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:208) 03-02
16:58:25.257: W/System.err(1868): at
java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
03-02 16:58:25.257: W/System.err(1868): at
java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
03-02 16:58:25.257: W/System.err(1868): at
java.lang.Thread.run(Thread.java:856)
Question
What I’m doing wrong and how I can solve this issue. Why I get “No peer certificate” exception ?
Thanks.
Edited
Windows Server settings.
<VirtualHost *:443>
ServerName 10.2.20.20
Alias /fido/EzPay/ "d:/fido/EzPay/"
Alias /fido/EzPay "d:/fido/EzPay/"
<Directory "d:/fido/EzPay/">
Options Indexes FollowSymLinks MultiViews
AllowOverride all
Order allow,deny
Allow from all
</Directory>
# These are the actual SSL directives needed to get it all working!
SSLEngine on
SSLCertificateFile C:/wamp/bin/apache/apache2.2.17/conf/ssl/fidoserver.crt
SSLCertificateKeyFile C:/wamp/bin/apache/apache2.2.17/conf/ssl/fidoserver.pem
</VirtualHost>
Finally I have solved https problem. As I fought the main problem was in server, concretely in certificate.
Android supports only “BKS” certificate and that’s was the reason that we can’t get response from the
server. In order to solve this issue I have read more then 30 articles and finally found solution.
The steps which I done to solve this issue you can see below:
First thing that I do was generating .bks keystore file from our fidoserver.crt certificate, in order to do that I have read this article and do following:
Before running this command I have download Bouncy Castle .jar file and put it in the folder with certificates. After doing this all steps I get keystore.bks file which is the right certificate file for Android application. I put this file in Androids mnc/sdcard folder. In java code I have write following code to read that keystore.bbk file
This all allow m to load our certificate with given password 222222 (password we give when create a keystore with keytool).
After this all my test application start to work correctly. Now I can send request to https and get response from it. I have tested
application with FIDO server, everything works great! I think on Monday I will make some changes in EzPay application and it
will start working with https connections.
References