- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试使用 Retrofit 实现摘要式身份验证。我的第一个解决方案在 OkHttpClient 上设置了 OkHttp 的 Authenticator 的实现:
class MyAuthenticator implements Authenticator {
private final DigestScheme digestScheme = new DigestScheme();
private final Credentials credentials = new UsernamePasswordCredentials("user", "pass");
@Override public Request authenticate(Proxy proxy, Response response) throws IOException {
try {
digestScheme.processChallenge(new BasicHeader("WWW-Authenticate", response.header("WWW-Authenticate")));
HttpRequest request = new BasicHttpRequest(response.request().method(), response.request().uri().toString());
String authHeader = digestScheme.authenticate(credentials, request).getValue();
return response.request().newBuilder()
.addHeader("Authorization", authHeader)
.build();
} catch (Exception e) {
throw new AssertionError(e);
}
}
@Override public Request authenticateProxy(Proxy proxy, Response response) throws IOException {
return null;
}
}
这非常适用于通过 Retrofit 进行的 GET 请求。但是,如 this StackOverflow question 中所述, POST 请求导致“无法重试流式传输 HTTP 正文”异常:
Caused by: java.net.HttpRetryException: Cannot retry streamed HTTP body
at com.squareup.okhttp.internal.http.HttpURLConnectionImpl.getResponse(HttpURLConnectionImpl.java:324)
at com.squareup.okhttp.internal.http.HttpURLConnectionImpl.getResponseCode(HttpURLConnectionImpl.java:508)
at com.squareup.okhttp.internal.http.HttpsURLConnectionImpl.getResponseCode(HttpsURLConnectionImpl.java:136)
at retrofit.client.UrlConnectionClient.readResponse(UrlConnectionClient.java:94)
at retrofit.client.UrlConnectionClient.execute(UrlConnectionClient.java:49)
at retrofit.RestAdapter$RestHandler.invokeRequest(RestAdapter.java:357)
at retrofit.RestAdapter$RestHandler.invoke(RestAdapter.java:282)
at $Proxy3.login(Native Method)
at com.audax.paths.job.LoginJob.onRunInBackground(LoginJob.java:41)
at com.audax.library.job.AXJob.onRun(AXJob.java:25)
at com.path.android.jobqueue.BaseJob.safeRun(BaseJob.java:108)
at com.path.android.jobqueue.JobHolder.safeRun(JobHolder.java:60)
at com.path.android.jobqueue.executor.JobConsumerExecutor$JobConsumer.run(JobConsumerExecutor.java:172)
at java.lang.Thread.run(Thread.java:841)
Jesse Wilson 解释说我们不能在验证后重新发送我们的请求,因为 POST 正文已经被丢弃了。但是由于摘要认证,我们需要返回的WWW-Authenticate
header ,所以我们不能使用RequestInterceptor
来简单地添加一个 header 。也许可以在 RequestInterceptor
中执行单独的 HTTP 请求,并在响应中使用 WWW-Authenticate
header ,但这看起来很老套。
有什么办法解决这个问题吗?
最佳答案
作为一种解决方法,我最终用 Apache 的 HttpClient 替换了 OkHttp,它具有内置的摘要身份验证。提供 retrofit.client.Client 的实现,将其请求委托(delegate)给 Apache 的 HttpClient:
import retrofit.client.Client;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.auth.Credentials;
import org.apache.http.auth.AuthScope;
import org.apache.http.client.CredentialsProvider;
import org.apache.http.impl.client.BasicCredentialsProvider;
import org.apache.http.impl.client.HttpClientBuilder;
import retrofit.client.Request;
import retrofit.client.Response;
public class MyClient implements Client {
private final CloseableHttpClient delegate;
public MyClient(String user, String pass, String hostname, String scope) {
Credentials credentials = new UsernamePasswordCredentials(user, pass);
AuthScope authScope = new AuthScope(hostname, 443, scope);
CredentialsProvider credentialsProvider = new BasicCredentialsProvider();
credentialsProvider.setCredentials(authScope, credentials);
delegate = HttpClientBuilder.create()
.setDefaultCredentialsProvider(credentialsProvider)
.build();
}
@Override public Response execute(Request request) {
//
// We're getting a Retrofit request, but we need to execute an Apache
// HttpUriRequest instead. Use the info in the Retrofit request to create
// an Apache HttpUriRequest.
//
String method = req.getMethod();
ByteArrayOutputStream bos = new ByteArrayOutputStream();
if (request.getBody() != null) {
request.getBody().writeTo(bos);
}
String body = new String(bos.toByteArray());
HttpUriRequest wrappedRequest;
switch (method) {
case "GET":
wrappedRequest = new HttpGet(request.getUrl());
break;
case "POST":
wrappedRequest = new HttpPost(request.getUrl());
wrappedRequest.addHeader("Content-Type", "application/xml");
((HttpPost) wrappedRequest).setEntity(new StringEntity(body));
break;
case "PUT":
wrappedRequest = new HttpPut(request.getUrl());
wrappedRequest.addHeader("Content-Type", "application/xml");
((HttpPut) wrappedRequest).setEntity(new StringEntity(body));
break;
case "DELETE":
wrappedRequest = new HttpDelete(request.getUrl());
break;
default:
throw new AssertionError("HTTP operation not supported.");
}
//
// Then execute the request with `delegate.execute(uriRequest)`.
//
// ...
//
}
然后在您的 RestAdapter.Builder 上设置新的客户端实现:
RestAdapter restAdapter = new RestAdapter.Builder()
.setClient(new MyClient("jason", "pass", "something.com", "Some Scope"))
.setEndpoint("https://something.com/api")
.build();
关于android - 使用摘要式 HTTP 身份验证 : “Cannot retry streamed HTTP body” 改造 POST 请求,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26613933/
我正在使用 Retrofit 和 RxJava,但似乎无法做我想做的事。 这是我对 Web 服务的声明: Observable rawRemoteDownload(@Header("Cookie")
我正在开发一个android应用程序,并使用改造发送请求和获取响应。像这样: public interface Service { @POST("/GetInfo") InfoResp
我刚刚从响应中获取代码,它说我的请求参数错误,那么我的 api 调用应该是什么样子? 这是来自文档的硬编码 API 调用 https://api.themoviedb.org/3/discover/m
实际上,当我使用具有 radio 频率的设备时,我的应用程序出现了问题。如果设备超出 radio 范围但尝试发送文件,我会收到 onFailure 消息,我对用户说没有网络,但主要问题是,一旦设备返回
我一直在关注其他答案,但缺少一个我找不到的步骤,这导致调用成功但数据未被正确解析,因为我进行的第一次调用返回了一个对象列表,但只返回了一个对象全部为空 我的模型.java public cla
对于我正在使用的服务器,我们有一个子域和一个目录,它们都绑定(bind)在一起。使用 Retrofit,您需要指定 baseURL,它似乎不允许目录。有什么方法可以实现吗? 例子: https://d
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题吗? 通过 editing this post 添加细节并澄清问题. 关闭 6 年前。 Improve t
我对 Android 还很陌生.. 我正在尝试使用 Retrofit 从 MySQL 检索数据.. 我在代码中没有发现任何错误,但是当我从设备运行应用程序时,它卡在“正在获取数据的进度对话框”上...
我正在使用 Dagger2 + Retrofit + RxAndroid + OkHttp3 + 新架构组件开发一个 Android 应用程序。 最小 sdk = 16。 问题:在 API 16 上运
我需要在插入硬编码查询后设置查询。 我的 API 地址是: myapiaddress/names?q=Yoni&gender=Man&(here i need to enter dynamic
我正在创建天气应用程序,让用户可以选择按任何城市搜索天气。当用户输入有效的城市名称时,我的应用程序工作正常,但当用户输入无效的字符串 ex: asndfs,bdfbsj 然后我的应用程序终止。 如何处
我有一个 Json 字符串。我无法使用带有两个列表(类别和产品)的 Retrofit onResponse。我怎么打电话,回调?我应该用什么?通常不是列表,对吧? { "Categories": [{
我正在实现一个两级嵌套的 recyclerView 并且两个回收器 View 都使用 retrofit 进行 API 调用。这是发出同步请求的方法: public void loadSectionSt
我正在使用 Qt 编写一个应用程序并且想要一个“Metro 风格”的界面。一切都完成了,除了我不知道如何让小部件出现和消失。例如,在 WPF 中,您可以为 (UIElement.RenderTrans
我在 java 应用程序中使用 Retrofit 来访问 api-rest。 我使用简单的代码: RestAdapter.Builder().setEndpoint(uri).setLogLevel(
我需要将下面报告的数据字符串转换为以时间戳(第一个数字元素)为键的字典。我该怎么做? element=20151201091000|22844.4|22786.2|22801.6|22839.7|10
我正在尝试使用 @QueryMap 发送多个参数(就像我通常做的那样)但是这次使用改造通过 POST。 改造 API @POST("/request.php") void sendRequest(@Q
我正在使用 Retrofit 进行后端通信:如果状态码不是 200 则回调调用失败方法。但是我想在失败方法中获取状态代码以进行进一步的代码调节 @Override pu
我正在使用自定义记录器记录到 Logcat 和文件(当文件启用时)。 这在接收来自测试人员的错误报告时非常有用(因为我在应用程序中还有一个按钮可以发送错误报告并附上日志)。 问题:我正在使用 Retr
很抱歉,如果我的标题含糊不清,但我找不到更好的。 我有一个以这种方式公开服务的休息 Api:/api/{type}/{id} 4 个“类型”,因此返回 4 个类类型。 所有这些类都继承自同一个父类(s
我是一名优秀的程序员,十分优秀!