ProtocolException: unexpected end of stream

MatejS picture MatejS · Apr 6, 2016 · Viewed 9.7k times · Source

I've been dealing with an issue that keeps coming back and I'm getting quite frustrated. I'm trying to connect to my database via a template php webservice that I found online, by sending it JSONObjects. I keep getting either an IOException or ProtocolException when trying to get a response code from the server with the cause being "Unexpected end of stream". The error message is really cryptic and I have no idea is the error is in the java or the PHP code. I posted the code, so you can take a look:

Stack trace:

JSONParser.java (I marked the offending line with a comment)

import android.util.Log;

import org.json.JSONException;
import org.json.JSONObject;

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.URL;


public class JSONParser {

    private JSONObject jsonObj = new JSONObject();
    private JSONObject json = new JSONObject();
    private String email;
    private String username;
    private String password;
    private int objLength;
    private static InputStream is = null;

    public JSONParser() {
        //Default constructor intentionally left empty for now
    }

    public JSONObject getJSONFromUrl(String _url, JSONObject params) {

        // Making HTTP request
        try {
            URL url = new URL(_url);
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();

            byte[] bytearray = params.toString().getBytes("UTF-8");

            conn.setDoOutput(true);
            conn.setDoInput(true);
            conn.setRequestMethod("POST");
            conn.setFixedLengthStreamingMode(bytearray.length);
            //conn.setRequestProperty("User-Agent", "GYUserAgentAndroid");
            conn.setRequestProperty("Content-Length", Integer.toString(bytearray.length));
            conn.setRequestProperty("Content-Type", "application/json");
            //conn.setUseCaches(false);

            System.out.println("Byte array length: "+bytearray.length);

            //The offending line                
            int responseCodeHTTP = conn.getResponseCode(); 

            System.out.println("Responsecode HTTP "+responseCodeHTTP);

            OutputStream os = conn.getOutputStream();
            os.write(bytearray);
            os.flush();

            if (responseCodeHTTP == HttpURLConnection.HTTP_OK) {
                try {
                    is = new BufferedInputStream(conn.getInputStream());
                    BufferedReader reader = new BufferedReader(new InputStreamReader(
                            is, "UTF-8"), 8);
                    StringBuilder sb = new StringBuilder();
                    String line = null;
                    while ((line = reader.readLine()) != null) {
                        sb.append(line);
                    }
                    is.close();
                    json = new JSONObject(sb.toString());
                    Log.e("JSON", json.toString());
                }
                catch (JSONException e) {
                    e.printStackTrace();
                }
            }

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

        return json;

    }

    public byte[] getJSONBytes() {
        try {
            return jsonObj.toString().getBytes("UTF-8");
        }
        catch(UnsupportedEncodingException e) {
            e.printStackTrace();
            return null;
        }
    }
}

index.php (I didn't change much in the code yet, just want to establish a connection for now)

<?php
 
if($_POST != null) {
	
	echo json_encode("Status: 200");
	
	if (isset($_POST['tag']) && $_POST['tag'] != '') {
    // Get tag
    $tag = $_POST['tag'];
 
    // Include Database handler
    require_once 'DB_Functions.php';
    $db = new DB_Functions();
    // response Array
    $response = array("tag" => $tag, "success" => 0, "error" => 0);
 
    // check for tag type
    if ($tag == 'login') {
        // Request type is check Login
        $email = $_POST['email'];
        $password = $_POST['password'];
 
        // check for user
        $user = $db->getUserByEmailAndPassword($email, $password);
		
		
		//FOR TESTING PURPOSES: REMOVE LATER
		$user = false;
		
		
        if ($user != false) {
            // user found
            // echo json with success = 1
            $response["success"] = 1;
            $response["user"]["fname"] = $user["firstname"];
            $response["user"]["lname"] = $user["lastname"];
            $response["user"]["email"] = $user["email"];
        $response["user"]["uname"] = $user["username"];
            $response["user"]["uid"] = $user["unique_id"];
            $response["user"]["created_at"] = $user["created_at"];
 
            echo json_encode($response);
        } else {
            // user not found
            // echo json with error = 1
            $response["error"] = 1;
            $response["error_msg"] = "Incorrect email or password!";
            echo json_encode($response);
        }
    }
  else if ($tag == 'chgpass'){
  $email = $_POST['email'];
 
  $newpassword = $_POST['newpas'];
 
  $hash = $db->hashSSHA($newpassword);
        $encrypted_password = $hash["encrypted"]; // encrypted password
        $salt = $hash["salt"];
  $subject = "Change Password Notification";
         $message = "Hello User,nnYour Password is sucessfully changed.nnRegards,nLearn2Crack Team.";
          $from = "[email protected]";
          $headers = "From:" . $from;
    if ($db->isUserExisted($email)) {
 
 $user = $db->forgotPassword($email, $encrypted_password, $salt);
if ($user) {
         $response["success"] = 1;
          mail($email,$subject,$message,$headers);
         echo json_encode($response);
}
else {
$response["error"] = 1;
echo json_encode($response);
}
 
            // user is already existed - error response
 
        }
           else {
 
            $response["error"] = 2;
            $response["error_msg"] = "User not exist";
             echo json_encode($response);
 
}
}
else if ($tag == 'forpass'){
$forgotpassword = $_POST['forgotpassword'];
 
$randomcode = $db->random_string();
 
$hash = $db->hashSSHA($randomcode);
        $encrypted_password = $hash["encrypted"]; // encrypted password
        $salt = $hash["salt"];
  $subject = "Password Recovery";
         $message = "Hello User,nnYour Password is sucessfully changed. Your new Password is $randomcode . Login with your new Password and change it in the User Panel.nnRegards,nLearn2Crack Team.";
          $from = "[email protected]";
          $headers = "From:" . $from;
    if ($db->isUserExisted($forgotpassword)) {
 
 $user = $db->forgotPassword($forgotpassword, $encrypted_password, $salt);
if ($user) {
         $response["success"] = 1;
          mail($forgotpassword,$subject,$message,$headers);
         echo json_encode($response);
}
else {
$response["error"] = 1;
echo json_encode($response);
}
 
            // user is already existed - error response
 
        }
           else {
 
            $response["error"] = 2;
            $response["error_msg"] = "User not exist";
             echo json_encode($response);
 
}
 
}
else if ($tag == 'register') {
        // Request type is Register new user
        $fname = $_POST['fname'];
        $lname = $_POST['lname'];
        $email = $_POST['email'];
        $uname = $_POST['uname'];
        $password = $_POST['password'];
 
        // check if user is already existed
        if ($db->isUserExisted($email)) {
            // user is already existed - error response
            $response["error"] = 2;
            $response["error_msg"] = "User already existed";
            echo json_encode($response);
        }
           else if(!$db->validEmail($email)){
            $response["error"] = 3;
            $response["error_msg"] = "Invalid Email Id";
            echo json_encode($response);
}
else {
            // store user
            $user = $db->storeUser($fname, $lname, $email, $uname, $password);
            if ($user) {
                // user stored successfully
            $response["success"] = 1;
            $response["user"]["fname"] = $user["firstname"];
            $response["user"]["lname"] = $user["lastname"];
            $response["user"]["email"] = $user["email"];
        $response["user"]["uname"] = $user["username"];
            $response["user"]["uid"] = $user["unique_id"];
            $response["user"]["created_at"] = $user["created_at"];
               mail($email,$subject,$message,$headers);
 
                echo json_encode($response);
            } else {
                // user failed to store
                $response["error"] = 1;
                $response["error_msg"] = "JSON Error occured in Registartion";
                echo json_encode($response);
            }
        }
    } else {
         $response["error"] = 3;
         $response["error_msg"] = "JSON ERROR";
        echo json_encode($response);
    }
} else {
    echo "";
}
}
 
?>

I really appreciate any help you can give. Especially because I'm still a noob in Android development and PHP.

Answer

Name is Nilay picture Name is Nilay · Feb 13, 2018

I struggled a lot for the solution of this issue. Searched various S/O posts nut I resolved the issue by adding header "Connection: close" in requests. I found solution here

Something like this will resolve the java.net.ProtocolException: unexpected end of stream:

okHttpClient = new OkHttpClient.Builder()
            .addNetworkInterceptor(new Interceptor() {
                @Override
                public Response intercept(Chain chain) throws IOException {
                    Request request = chain.request().newBuilder().addHeader("Connection", "close").build();
                    return chain.proceed(request);
                }
            })
            .build();