安卓:org.apache.http.NoHttpResponseException:目标服务器未响应

4

我的Android应用程序使用HttpClient/HttpGet访问REST API。我还设置了:

httpGet.addHeader("Authorization", "Basic " + basicAuth);

...该函数将Base64编码的“用户名:密码”发送到服务器。为了测试,我只使用HTTP而不是HTTPS连接到服务器。

每当我在HttpClient上调用“execute”时,都会收到以下异常:

07-28 10:51:39.610: I/# PA #(12112): org.apache.http.NoHttpResponseException: The target server failed to respond
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.conn.DefaultResponseParser.parseHead(DefaultResponseParser.java:85)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.io.AbstractMessageParser.parse(AbstractMessageParser.java:174)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.AbstractHttpClientConnection.receiveResponseHeader(AbstractHttpClientConnection.java:180)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.conn.DefaultClientConnection.receiveResponseHeader(DefaultClientConnection.java:235)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.conn.AbstractClientConnAdapter.receiveResponseHeader(AbstractClientConnAdapter.java:259)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.protocol.HttpRequestExecutor.doReceiveResponse(HttpRequestExecutor.java:279)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.protocol.HttpRequestExecutor.execute(HttpRequestExecutor.java:121)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.client.DefaultRequestDirector.execute(DefaultRequestDirector.java:428)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:555)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:487)
07-28 10:51:39.610: I/# PA #(12112):    at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:465)
07-28 10:51:39.610: I/# PA #(12112):    at org.thomasamsler.android.app.tasks.HttpGetTask.doInBackground(HttpGetTask.java:87)
07-28 10:51:39.610: I/# PA #(12112):    at org.foo.android.app.tasks.HttpGetTask.doInBackground(HttpGetTask.java:1)
07-28 10:51:39.610: I/# PA #(12112):    at android.os.AsyncTask$2.call(AsyncTask.java:264)
07-28 10:51:39.610: I/# PA #(12112):    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
07-28 10:51:39.610: I/# PA #(12112):    at java.util.concurrent.FutureTask.run(FutureTask.java:137)
07-28 10:51:39.610: I/# PA #(12112):    at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:208)
07-28 10:51:39.610: I/# PA #(12112):    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
07-28 10:51:39.610: I/# PA #(12112):    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
07-28 10:51:39.610: I/# PA #(12112):    at java.lang.Thread.run(Thread.java:856)

注意:

我实际上有三台使用不同托管提供商的开发服务器。连接其中两台时,我会收到上述错误。连接第三个服务器时,一切工作正常。此外,所有三个开发服务器上的其他GET和POST请求都可以正常工作。

因此,我想知道在设置基本身份验证头时,是否有与HTTP GET特定相关的内容。

我也可以使用“curl”调用相关的REST API而没有任何问题。

代码:

import java.io.ByteArrayOutputStream;
import java.io.IOException;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;

import android.os.AsyncTask;
import android.util.Log;

public class HttpGetTask extends AsyncTask<HttpGetRequestArgs, Void, String> implements AppConstants {

    private HttpTaskNotifier mNotifier;

    private HttpResponse mHttpResponse;
    private int mHttpStatusCodeOk;
    private int mHttpStatusCode;
    private int mErrorCode = AppConstants.NO_ERROR;

    public HttpGetTask(int httpStatusCodeOk, HttpTaskNotifier notifier) {

        mNotifier = notifier;
        mHttpStatusCodeOk = httpStatusCodeOk;
    }

    @Override
    protected String doInBackground(HttpGetRequestArgs... args) {

        HttpGetRequestArgs httpGetRequestArgs = args[0];

        HttpClient httpClient = new DefaultHttpClient();
        HttpGet httpGet = null;

        try {

            httpGet = new HttpGet(httpGetRequestArgs.getUrl());

            String apiKey = httpGetRequestArgs.getApiKey();

            if (null != apiKey && !"".equals(apiKey)) {

                httpGet.addHeader("Authorization", "ApiKey " + apiKey);
            }

            String basicAuth = httpGetRequestArgs.getBasicAuth();

            if (null != basicAuth && !"".equals(basicAuth)) {

                httpGet.addHeader("Authorization", "Basic " + basicAuth);
            }

        }
        catch (IllegalArgumentException e) {

            Log.e(LOG_TAG, "EXCEPTION", e);
            return null;
        }

        try {

            mHttpResponse = httpClient.execute(httpGet);
        }
        catch (ClientProtocolException e) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }
        catch (IOException e) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }
        catch (IllegalArgumentException iae) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }

        if (null == mHttpResponse) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }

        mHttpStatusCode = mHttpResponse.getStatusLine().getStatusCode();

        if (mHttpStatusCodeOk != mHttpStatusCode) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }

        ByteArrayOutputStream out = new ByteArrayOutputStream();
        HttpEntity httpEntity = mHttpResponse.getEntity();

        if (null == httpEntity) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }

        try {

            httpEntity.writeTo(out);
            out.close();
        }
        catch (IOException ioe) {

            mErrorCode = AppConstants.REST_ERROR;
            return null;
        }

        return out.toString();
    }

    @Override
    protected void onPostExecute(String content) {

        if (AppConstants.NO_ERROR != mErrorCode) {

            mNotifier.onError(mErrorCode);
        }

        mNotifier.doProcess(content);
    }

    @Override
    protected void onCancelled() {

        mNotifier.doCancel();
    }
}

我忘了提到以下内容:当异常发生时,看起来就像客户端实际上尝试连续4次执行REST调用。我在服务器上添加了一个调试语句,就在REST调用执行时,我看到调试消息连续出现了4次。 - tamsler
2个回答

8
我终于弄清楚问题所在了。当我对用户名和密码进行Base64编码时,我使用了错误的标志(Base64.encodeToString(...))。所以我更改了标志:

修改前:

basicAuth = Base64.encodeToString(basicAuth.getBytes(), Base64.DEFAULT);

目标(收件)人:
basicAuth = Base64.encodeToString(basicAuth.getBytes(), Base64.NO_WRAP);

"......一切都正常运作。"

我也遇到了类似的问题。但是我的错误在于我使用了http,而服务器地址是https。更改后解决了我的问题。 - SKT


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接