android httpclient works but my similar HttpUrlconnection does not post

626 Views Asked by At

As many of you know new apps should use HttpUrlconnection because Httpclient will not work past api 22. I have below code in HttpClient that successfully posts to the server and then I comment that out and swap in HttpUrlconnection and it does not post. What could possibly be wrong with my code for HttpUrlconnection

    URL url = new URL(LOGIN_URL);
        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
        conn.connect();
        conn.setReadTimeout(10000);
        conn.setConnectTimeout(15000);
        conn.setRequestMethod("POST");
       String cert="username=user_777&password=76566";
        conn.setDoOutput(true);
        DataOutputStream wr = new DataOutputStream(conn.getOutputStream());
        wr.writeBytes(cert);
        wr.flush();
        wr.close();

      conn.disconnect();

This similar code in HttpClient works perfectly

 HttpClient httpClient = new DefaultHttpClient();
        HttpPost httpPost = new HttpPost(LOGIN_URL);
        List<NameValuePair> nameValuePair = new ArrayList<NameValuePair>(2);
        nameValuePair.add(new BasicNameValuePair("username", "test_user"));
        nameValuePair.add(new BasicNameValuePair("password", "123456789"));
        httpPost.setEntity(new UrlEncodedFormEntity(nameValuePair));
        HttpResponse response = httpClient.execute(httpPost);
        Log.d("Http Post Response:", response.toString());

Httpclient works, HttpUrlclient does not how can I make it work? They both have the same URL.

1

There are 1 best solutions below

1
On BEST ANSWER

The problem is that you're calling conn.setDoOutput(true) when the connection is already open.

From the documentation:

Sets the flag indicating whether this URLConnection allows output. It cannot be set after the connection is established.

Moving the call to conn.setDoOutput(true); to before conn.connect(); fixed the issue.

Here is the error that I got with your original code:

 java.lang.RuntimeException: An error occured while executing doInBackground()
            at android.os.AsyncTask$3.done(AsyncTask.java:300)
            at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:355)
            at java.util.concurrent.FutureTask.setException(FutureTask.java:222)
            at java.util.concurrent.FutureTask.run(FutureTask.java:242)
            at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:231)
            at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
            at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
            at java.lang.Thread.run(Thread.java:841)
     Caused by: java.lang.IllegalStateException: Already connected
            at java.net.URLConnection.checkNotConnected(URLConnection.java:464)
            at java.net.URLConnection.setDoOutput(URLConnection.java:878)

Here is fully working and tested code, note I added some try/catch blocks and moved things around a bit, and also added code to read the result from the server:

class MyAsyncTask extends AsyncTask<String, String, String> {

    String LOGIN_URL = "http://www.example.com/someFile.php";
    HttpURLConnection conn;
    DataOutputStream wr;
    StringBuilder result = new StringBuilder();
    URL url;

    @Override
    protected String doInBackground(String... params) {

        try {
            url = new URL(LOGIN_URL);

            conn = (HttpURLConnection) url.openConnection();

            conn.setDoOutput(true); //moved here

            conn.connect();

        } catch (IOException e) {
            e.printStackTrace();
        }

        conn.setReadTimeout(10000);
        conn.setConnectTimeout(15000);
        try {
            conn.setRequestMethod("POST");
        } catch (ProtocolException e) {
            e.printStackTrace();
        }
        String cert="name=user_777&message=7656666666";


        try {
            wr = new DataOutputStream(conn.getOutputStream());
            wr.writeBytes(cert);
            wr.flush();
            wr.close();

            InputStream in = new BufferedInputStream(conn.getInputStream());
            BufferedReader reader = new BufferedReader(new InputStreamReader(in));

            String line;
            while ((line = reader.readLine()) != null) {
                result.append(line);
            }

        } catch (IOException e) {
            e.printStackTrace();
        }


        conn.disconnect();

        return result.toString();
    }

    @Override
    protected void onPostExecute(String result) {

        Toast.makeText(MainActivity.this, result, Toast.LENGTH_LONG ).show();
    }
}