Retrofit not retrieving AccessToken

2019-08-07 05:53发布

问题:

I have been following this tutorial as I have been advised that Retrofit is the best API to use for oAuth and network calls: Retrofit oAuth2 tutorial

It seems to have been working well and I had retrieved my authorization code without any problems. However, when It came to using the code they supplied for exchanging it for an Access Token, I seem to be getting a nullpointer and being a total beginner with Retrofit I am not sure what I am doing wrong.

ServiceGenerator Class:

public class ServiceGenerator {

public static final String API_BASE_URL = "https://xxxx.xxxx.com/oauth/";

private static OkHttpClient.Builder httpClient = new OkHttpClient.Builder();

private static Retrofit.Builder builder =
        new Retrofit.Builder()
                .baseUrl(API_BASE_URL)
                .addConverterFactory(GsonConverterFactory.create());

private static Retrofit retrofit = builder.build();

public static <S> S createService(Class<S> serviceClass) {
    return createService(serviceClass, null);
}

public static <S> S createService(
        Class<S> serviceClass, String clientId, String clientSecret) {
    if (!TextUtils.isEmpty(clientId)
            && !TextUtils.isEmpty(clientSecret)) {
        String authToken = Credentials.basic(clientId, clientSecret);
        return createService(serviceClass, authToken);
    }

    return createService(serviceClass, null, null);
}

public static <S> S createService(
        Class<S> serviceClass, final String authToken) {
    if (!TextUtils.isEmpty(authToken)) {
        AuthenticationInterceptor interceptor =
                new AuthenticationInterceptor(authToken);

        if (!httpClient.interceptors().contains(interceptor)) {
            httpClient.addInterceptor(interceptor);

            builder.client(httpClient.build());
            retrofit = builder.build();
        }
    }
    return retrofit.create(serviceClass);
}

}

AuthenticationInterceptor Class:

public class AuthenticationInterceptor implements Interceptor {

private String authToken;

public AuthenticationInterceptor(String token) {
    this.authToken = token;
}

@Override
public Response intercept(Chain chain) throws IOException {
    Request original = chain.request();

    Request.Builder builder = original.newBuilder()
            .header("Authorization", authToken);

    Request request = builder.build();
    return chain.proceed(request);
}

}

Access Token Class:

public class AccessToken {

private String accessToken;
private String tokenType;

public String getAccessToken() {
    return accessToken;
}

public String getTokenType() {
    // OAuth requires uppercase Authorization HTTP header value for token type
    if (! Character.isUpperCase(tokenType.charAt(0))) {
        tokenType =
                Character
                        .toString(tokenType.charAt(0))
                        .toUpperCase() + tokenType.substring(1);
    }

    return tokenType;
}

}

Most importantly, my oAuth Class:

public class oAuthAuthentication extends Activity {
// you should either define client id and secret as constants or in string resources
private final String clientId = "xxxxxxxxxxxxxxx";
private final String clientSecret = "xxxxxxx";
private final String redirectUri = "https://xxxx.xxxx.xxx/rest/callback.html";
private String AUTH_URL;
private String AuthCode;
String grantType;
AccessToken accessToken;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.oauth);

    Button loginButton = (Button) findViewById(R.id.login);
    loginButton.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            String url = AUTH_URL;
            Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
            startActivity(intent);
        }
    });
}

@Override
protected void onResume() {
    super.onResume();

    // the intent filter defined in AndroidManifest will handle the return from ACTION_VIEW intent
    Uri uri = getIntent().getData();
    if (uri != null && uri.toString().startsWith(redirectUri)) {
            new AsyncTaskRunner().execute();
        } else {
            Log.e("ERROR", "Error has occured with AuthCode");
        }
    }




public interface LoginService {
        @FormUrlEncoded
        @POST("/oauth/token")
        Call<AccessToken> getAccessToken(
                @Field("code") String code,
                @Field("grant_type") String grantType,
                @Field("client_id") String clientID,
                @Field("client_secret") String clientSecret,
                @Field("redirect_uri") String redirectUri);

    }

private class AsyncTaskRunner extends AsyncTask<String, String, String> {
    @Override
    protected String doInBackground(String... params) {
        Uri uri = getIntent().getData();
        if (uri != null && uri.toString().startsWith(redirectUri)) {
            // use the parameter your API exposes for the code (mostly it's "code")
            String code = uri.getQueryParameter("code");
            if (code != null) {
                AuthCode = code;
                LoginService loginService =
                        ServiceGenerator.createService(LoginService.class, clientId, clientSecret);
                Call<AccessToken> call = loginService.getAccessToken(code, "authorization_code");
                try {
                    accessToken = call.execute().body();
                    Log.e("token", accessToken.toString());
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return String.valueOf(accessToken);
    }
}

}

The AccessToken End-point I am trying to hit is https://xxxx.xxxx.com/oauth/token

Is there anyone who has used Retrofit for oAuth can help me with this? P.s. I have checked all my values are correct, ID, Secret, Code is being generated fine and grant_type is supposed to equal "authorization_code" as stated.