Я следовал этому руководству, так как мне сообщили, что Retrofit
- лучший API для использования в oAuth и сетевых вызовах:
Руководство по модернизации oAuth2
Кажется, все работало хорошо, и я без проблем получил свой код авторизации. Однако, когда дело дошло до использования кода, который они предоставили для замены его на токен доступа, я, похоже, получил nullpointer
и был полным новичком с Retrofit
Я не уверен, что я делаю неправильно.
Класс ServiceGenerator:
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:
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);
}
}
Класс токена доступа:
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;
}
}
Самое главное, мой класс oAuth:
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);
}
}
}
Конечная точка AccessToken, которую я пытаюсь достичь, - https://xxxx.xxxx.com/oauth/token
Кто-нибудь, кто использовал Retrofit для oAuth, может мне помочь с этим?
Постскриптум Я проверил, что все мои значения верны, ID, Секрет, Код генерируется нормально, а grant_type должен равняться "authorization_code", как указано.