我正在使用概述here的“令牌检索(代码流)”来通过OAuth2为Reddit Api检索访问令牌。我设法获取“代码”字符串,但在尝试检索访问令牌时遇到了unsupported_grant_type错误。我想使用Retrofit2来实现这一目标。
private void getAccessToken(String code) {
if (mRetrofit == null) {
mRetrofit = new Retrofit.Builder()
.baseUrl(RedditConstants.REDDIT_BASE_URL)
.addConverterFactory(GsonConverterFactory.create())
.build();
}
Api api = mRetrofit.create(Api.class);
String authString = RedditConstants.REDDIT_CLIENT_ID + ":";
String encodedAuthString = Base64.encodeToString(authString.getBytes(), Base64.NO_WRAP);
Map<String, String> headers = new HashMap<>();
// headers.put("Content-Type", "application/x-www-form-urlencoded");
// headers.put("Accept", "application/json");
headers.put("Authorization", "Basic " + encodedAuthString);
headers.put("grant_type" , "authorization_code");
headers.put("code", code);
headers.put("redirect_uri", RedditConstants.REDDIT_REDIRECT_URL);
headers.put("User-Agent", RedditConstants.REDDIT_USER_AGENT);
Call<RedditAccessToken> call = api.login(headers, MediaType.parse("application/x-www-form-urlencoded"));
call.enqueue(new Callback<RedditAccessToken>() {
@Override
public void onResponse(Call<RedditAccessToken> call, Response<RedditAccessToken> response) {
Log.d("Findme", "body: " + response.body().toString());
Log.d("Findme", "response: " + response.toString());
}
@Override
public void onFailure(Call<RedditAccessToken> call, Throwable t) {
Log.e("Fineme", "onFailure: " + t.getMessage());
}
});
}
记录结果:
D/Findme: body: RedditAccessToken{scope='null', token_type='null',
error='unsupported_grant_type', expires_in='0', access_token='null',
refresh_token='null'}
D/Findme: response: Response{protocol=h2, code=200, message=,
url=https://www.reddit.com/api/v1/access_token/}
改造界面:
public interface Api {
@POST("access_token/")
Call<RedditAccessToken> login (
@HeaderMap Map<String, String> headers,
@Header("Content-Type") MediaType contentType
);
}
改造数据模型:
public class RedditAccessToken {
@SerializedName("scope")
@Expose
private String scope;
@SerializedName("token_type")
@Expose
private String token_type;
@SerializedName("expires_in")
@Expose
private long expires_in;
@SerializedName("access_token")
@Expose
private String access_token;
@SerializedName("refresh_token")
@Expose
private String refresh_token;
@SerializedName("error")
@Expose
private String error;
public String getScope() {
return scope;
}
public String getRefresh_token() {
return refresh_token;
}
public void setRefresh_token(String refresh_token) {
this.refresh_token = refresh_token;
}
public void setScope(String scope) {
this.scope = scope;
}
public String getToken_type() {
return token_type;
}
public void setToken_type(String token_type) {
this.token_type = token_type;
}
public long getExpires_in() {
return expires_in;
}
public void setExpires_in(long expires_in) {
this.expires_in = expires_in;
}
public String getAccess_token() {
return access_token;
}
public void setAccess_token(String access_token) {
this.access_token = access_token;
}
@Override
public String toString() {
return "RedditAccessToken{" +
"scope='" + scope + '\'' +
", token_type='" + token_type + '\'' +
", error='" + error + '\'' +
", expires_in='" + expires_in + '\'' +
", access_token='" + access_token + '\'' +
", refresh_token='" + refresh_token + '\'' +
'}';
}
public String getError() {
return error;
}
public void setError(String error) {
this.error = error;
}
}
答案 0 :(得分:2)
我已经解决了错误,现在一切正常。首先,我应该使用Retrofit Annotation @FormUrlEncoded,它自动将mime类型调整为application / x-www-form-urlencoded。此外,只需要HTTP Basic Auth String标头。其他POST数据应作为Retrofit Fields提交。这些更改导致以下代码。
String authString = RedditConstants.REDDIT_CLIENT_ID + ":";
String encodedAuthString =
Base64.encodeToString(authString.getBytes(), Base64.NO_WRAP);
Map<String, String> headers = new HashMap<>();
headers.put("Authorization", "Basic " + encodedAuthString);
Map<String, String> fields = new HashMap<>();
fields.put("grant_type", "authorization_code");
fields.put("code", code);
fields.put("redirect_uri", RedditConstants.REDDIT_REDIRECT_URL);
fields.put("User-Agent", RedditConstants.REDDIT_USER_AGENT);
Call<RedditAccessToken> call = api.login(headers, fields);
call.enqueue(new Callback<RedditAccessToken>()
改造界面:
public interface Api {
@FormUrlEncoded
@POST("access_token/")
Call<RedditAccessToken> login (
@HeaderMap Map<String, String> headers,
@FieldMap Map<String, String> fields
);
}
答案 1 :(得分:0)
大!!!谢谢@ zachery-osborn
这项工作对我来说:D首先我有不支持_grant_type错误
我的服务器是ASP WebAPI 2 Owin OAuthAuthorizationServerProvider用于在登录中获取UserToken
获取令牌示例WebAPI的网址:http://localhost:41155/token
内容类型:应用/ X WWW的窗体-urlencoded
BulkEdit Mode Postman中的正文示例:
用户名:admin
密码:admin
grant_type:密码
响应: { “的access_token”: “wZ8Q54Mxpaq .....”, “token_type”: “承载”, “expires_in”:86399}
在Retrofit API接口中:
@FormUrlEncoded
@POST("token")
Call<ResponseBody> GetUserToken(@FieldMap Map<String, String> fields);
活动
Map<String, String> fields = new HashMap<>();
fields.put("username", "parsian");
fields.put("password", "admin");
fields.put("grant_type", "password");RetrofitInitialize.With(MainActivity.this).webServices.GetUserToken(fields);