OkHttp3 发送Digest Auth摘要认证

本文讲述了在使用OkHttp3的DigestAuth进行认证时遇到循环请求过多的问题,通过解析第一次请求的WWW-Authenticate头信息,自定义二次请求的Authorization头以避免重复发送,从而解决了问题。
摘要由CSDN通过智能技术生成

注意OkHttp3 自带的Digest Auth 认证存在循环请求:部分请求如下所示:

private static void digestAuth(ApiRequestVo apiRequestVo,Request.Builder requestBuilder,OkHttpClient.Builder okHttpClientBuilder){
        List<Map<String, Object>> authParams = apiRequestVo.getAuthParams();
        Map<String,String> map = new HashMap<>();
        for (Map<String, Object> authParam : authParams) {
            String value = String.valueOf(authParam.get(Params.VALUE)).trim();
            if (StringUtils.isNotEmpty(value)){
                map.put(String.valueOf(authParam.get(Params.NAME)).trim(),value);
            }
        }
        if (map.size() == 2){
            //这里需要来两次请求
            String username = map.get(Params.USERNAME);
            String password = map.get(Params.PASSWORD);
            okHttpClientBuilder.retryOnConnectionFailure(true)
                    .connectTimeout(15, TimeUnit.SECONDS)
                    .authenticator(new DigestAuthenticator(username,password))
                    .followRedirects(false);
        }
    }

如果这么传入会出现 Too many follow-up requests: 21这个错误。根据Digest Auth自身请求规则,第一次是客户端传过来的认证key,第二次根据key继续请求。
详细数据如下:
第一次:
在这里插入图片描述

第二次:
在这里插入图片描述

那么根据两次发送原则我们进行分解,由我们自己操控:

    @Test
    public void digestAuth(){
        OkHttpClient.Builder okHttpClientBuilder = new OkHttpClient.Builder();
        Request.Builder requestBuilder = new Request.Builder();
        String URL = "http://127.0.0.1:8082/digestAuth/info";
        requestBuilder.url(URL);
        Request request = requestBuilder.build();
        OkHttpClient client = okHttpClientBuilder.build();
        ApiResult apiResult = null;
        try {
            Response response = client.newCall(request).execute();
            String header = response.header("WWW-Authenticate");
            System.out.println(response);

            DigestScheme mDigestScheme = new DigestScheme();
            UsernamePasswordCredentials mCredentials = new UsernamePasswordCredentials("admin", "admin");
            mDigestScheme.processChallenge(new BasicHeader("WWW-Authenticate", response.header("WWW-Authenticate")));
            org.apache.http.HttpRequest request111 = new BasicHttpRequest(
                    response.request().method(),
                    response.request().url().toString());
            String authHeader;
            authHeader = mDigestScheme.authenticate(mCredentials, request111).getValue();

            requestBuilder.header("Authorization",authHeader);
            Response response1 = client.newCall(requestBuilder.build()).execute();
            System.out.println(response1);



        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (MalformedChallengeException e) {
            throw new RuntimeException(e);
        } catch (AuthenticationException e) {
            throw new RuntimeException(e);
        }
    }

自己根据第一次请求里response里的WWW-Authenticate,反写回第二次请求的Request header中,那么就可以规避循环发送问题。

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用okhttp3发送POST请求可以通过以下步骤实现: 1. 首先,在你的项目中添加okhttp3的依赖。可以在build.gradle文件中添加以下代码引入okhttp3库: implementation 'com.squareup.okhttp3:okhttp:3.7.0' implementation 'com.squareup.okio:okio:1.13.0' 2. 然后,创建OkHttpClient对象,并设置连接超时、读取超时和写入超时的时间。可以使用以下代码创建OkHttpClient对象: OkHttpClient client = new OkHttpClient().newBuilder() .connectTimeout(180, TimeUnit.SECONDS) .readTimeout(180, TimeUnit.SECONDS) .writeTimeout(180, TimeUnit.SECONDS) .build(); 3. 接下来,构建请求参数并创建RequestBody对象。可以使用FormBody.Builder()来添加参数。例如,以下代码创建了一个包含userName参数的RequestBody对象: FormBody body = new FormBody.Builder() .add("userName", "admin") .build(); 4. 然后,构建Request对象,设置请求的URL和请求方法为POST,并将RequestBody添加到请求中。例如,以下代码创建了一个POST请求的Request对象: String url = "http://example.com/api"; // 请求的URL Request request = new Request.Builder() .url(url) .post(body) .build(); 5. 最后,使用client的newCall方法执行请求,并通过Response对象获取响应结果。以下是一个简单的示例代码: try { Response response = client.newCall(request).execute(); String result = response.body().string(); // 处理响应结果 } catch (IOException e) { // 处理异常 } 这样就可以使用okhttp3发送POST请求了。请注意,以上示例中的代码仅作为参考,具体实现方式可能需要根据你的项目需求进行调整。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [okhttp3使用post请求](https://blog.csdn.net/as425017946/article/details/107321748)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [使用OkHttp3发起POST或GET请求](https://blog.csdn.net/Crezfikbd/article/details/128473202)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值