使用Retrofit(java)的Reddit Api unsupported_grant_type错误

时间:2017-12-13 14:25:27

标签: java android api retrofit reddit

我正在使用概述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;
    }
}

2 个答案:

答案 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);