Retrofit2发送承载令牌

时间:2020-04-07 15:50:07

标签: android retrofit2

我尝试发送带有HeaderMap的令牌,但得到401代码响应。设置项目的方式是,我有一个ApiClient单独的文件,并且有一个OkHttpClient Interceptor和一个HttpLoggingInterceptor来查看发生了什么,但是我无法获得Bearer令牌起作用。我已经看到了将其作为拦截器中的标头添加到拦截器中的解决方案,并且我已经尝试过了,但是由于我的令牌已保存在SharedPreferences中,因此我无法在拥有的ApiClient类中使用它。

这是ApiClient

HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();
interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);

Gson gson = new GsonBuilder().serializeNulls().setLenient().create();

OkHttpClient okHttpClient = new OkHttpClient.Builder()
        .addInterceptor(new Interceptor() {
            @NotNull
            @Override
            public okhttp3.Response intercept(@NotNull Chain chain) throws IOException {
                Request originalRequest = chain.request();
                Request newRequest = originalRequest.newBuilder()
                //I would add the header here
                //I tried this but it says on "ApiClient.this" cannot be referenced from static context
                // .header("Authorization" , SharedPreferencesHelper.getUserToken(ApiClient.this));
                        .build();

                return chain.proceed(newRequest);
            }
        })
        .addInterceptor(interceptor)
        .build();

retrofit = new Retrofit.Builder()
        .baseUrl("http://192.168.0.6:8000/api/")
        .addConverterFactory(GsonConverterFactory.create())
        .client(okHttpClient)
        .build();

这是SharedPreferencesHelper.getUserToken(MainActivity.this)

中的方法
public static String getUserToken(Context context) {
    SharedPreferences sp = getSharedPreferences(context);
    return sp.getString(USER_TOKEN, null);
}

这是当前调用,响应为401,如果我不添加Accept => application / json,则响应URL不正确,并且当我需要简单响应return response("LoggedOut", 200); //this is the response in the api时还返回html页面< / p>

Map<String, String> headers = new HashMap<>();
headers.put("Accept", "application/json");
headers.put("Token", SharedPreferencesHelper.getUserToken(MainActivity.this));

Call<Void> call = apiInterface.LogoutUser(headers);

call.enqueue(new Callback<Void>() {
    // onResponse and onFailure here
}

例如,如果没有Accept标头,这就是Logcat中的响应

D/OkHttp: --> GET http://192.168.0.6:8000/api/logout
D/OkHttp: Token: wE1Y8IxJpwyXtvw0fYoXZAlQ6qCx24YtzonQIeJBQSHmNppe0Sn1kLYDgZKCw4MKbpab4Vspf61Nzer1
D/OkHttp: --> END GET
D/OkHttp: <-- 200 OK http://192.168.0.6:8000/login
//a bunch of html that's the web page at this route, notice the /api is missing 

如何正确发送此邮件?

编辑: 我在后端使用Laravel项目,这是相关的路线

Route::middleware('auth:sanctum')
    ->get('/logoutApi', function (Request $request) {

    $request->user()->tokens()->delete();

    return response("LoggedOut", 202);
});

1 个答案:

答案 0 :(得分:0)

创建类Authenticator,例如:

const val HEADER_TOKEN_FIELD = "Authorization"
class ClassAuthenticator(
    private val pref: SharedPref
) : Authenticator {
    override fun authenticate(route: Route?, response: Response): Request? {

        return response.request().newBuilder()
                .header(HEADER_TOKEN_FIELD, pref.getToken())
                .build()
    }
}

然后使用以下方法在您的客户端中添加拦截器:

 val httpClient = OkHttpClient.Builder()
            .authenticator(ClassAuthenticator(pref))
            .addInterceptor { chain ->
                val request = chain.request()
                val httpRequest = request.newBuilder()
                        .addHeader(HEADER_TOKEN_FIELD, 
                          "Bearer ${pref.getToken()}")
                        .build()
                val response = chain.proceed(httpRequest)
                response
            }
            .build()
相关问题