我正在使用改造为Reddit创建一个Android客户端。 在logcat上获取toke工作正常,但是当我试图获取登录用户的信息时,我得到了#34; 404 Not Found"。 这是我的logcat:
D/OkHttp: --> POST https://oauth.reddit.com/api/v1/me http/1.1
D/OkHttp: Content-Length: 0
D/OkHttp: Authorization: bearer myToken
D/OkHttp: User-Agent: MyRedditClient/0.1 by myusername
D/OkHttp: --> END POST
D/OkHttp: <-- 404 Not Found https://oauth.reddit.com/api/v1/me (677ms)
这是我的代码:
import java.io.IOException;
import okhttp3.Interceptor;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.logging.HttpLoggingInterceptor;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class ServiceAuthGenerator {
public static final String API_BASE_URL = "https://www.reddit.com/";
private static OkHttpClient.Builder httpClient = new OkHttpClient.Builder();
private static Retrofit.Builder builder =
new Retrofit.Builder()
.baseUrl(API_BASE_URL)
.addConverterFactory(GsonConverterFactory.create());
public static <S> S getTokenService(Class<S> serviceClass) {
return getTokenService(serviceClass, null, null);
}
public static <S> S getTokenService(Class<S> serviceClass, String username, String password) {
if (username != null && password != null) {
String credentials = username + ":" + password;
final String basic =
"Basic " + Base64.encodeToString(credentials.getBytes(), Base64.NO_WRAP);
httpClient.addInterceptor(new Interceptor() {
@Override
public Response intercept(Interceptor.Chain chain) throws IOException {
Request original = chain.request();
Request.Builder requestBuilder = original.newBuilder()
.header("Authorization", basic)
.header("Accept", "application/json")
.method(original.method(), original.body());
Request request = requestBuilder.build();
return chain.proceed(request);
}
});
}
HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();
interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
OkHttpClient client = httpClient.addInterceptor(interceptor).build();
Retrofit retrofit = builder.client(client).build();
return retrofit.create(serviceClass);
}
}
import java.io.IOException;
import okhttp3.Interceptor;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.logging.HttpLoggingInterceptor;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class ServiceInfoGenerator {
public static final String API_BASE_URL = "https://oauth.reddit.com/";
private static OkHttpClient.Builder httpClient = new OkHttpClient.Builder();
private static Retrofit.Builder builder =
new Retrofit.Builder()
.baseUrl(API_BASE_URL)
.addConverterFactory(GsonConverterFactory.create());
public static <S> S retrieveInfoService(Class<S> serviceClass) {
return retrieveInfoService(serviceClass, null, null);
}
public static <S> S retrieveInfoService(Class<S> serviceClass, final String authToken, final String username) {
if (authToken != null && username != null) {
httpClient.addInterceptor(new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request original = chain.request();
// Request customization: add request headers
Request.Builder requestBuilder = original.newBuilder()
.header("Authorization", " bearer "+authToken)
.header("User-Agent", "MyRedditClient/0.1 by "+username)
.method(original.method(), original.body());
Request request = requestBuilder.build();
return chain.proceed(request);
}
});
}
HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();
interceptor.setLevel(HttpLoggingInterceptor.Level.HEADERS);
OkHttpClient client = httpClient.addInterceptor(interceptor).build();
try {
client.interceptors().add(new UserAgentInterceptor(username));
} catch (Exception e){
Log.i("ServiceInfoGenerator", "retrieveInfoService: "+e.getMessage());
}
Retrofit retrofit = builder.client(client).build();
return retrofit.create(serviceClass);
}
}
import java.util.Map;
import retrofit2.Call;
import retrofit2.http.Field;
import retrofit2.http.FormUrlEncoded;
import retrofit2.http.GET;
import retrofit2.http.Headers;
import retrofit2.http.POST;
import retrofit2.http.QueryMap;
public interface MyApiRetrofit {
@POST("/api/v1/access_token")
@FormUrlEncoded
Call<Authorize> accessToken(@Field(("grant_type")) String grantType, @Field(("username")) String username, @Field(("password")) String password);
@POST("/api/v1/me")
Call<Authorize> retrieveMyInfo();
}
import android.os.AsyncTask;
import android.util.Log;
import com.havistudio.myreddit.api.Authorize;
import com.havistudio.myreddit.api.MyApiRetrofit;
import java.io.IOException;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
public class TestAPITask2 extends AsyncTask<Void, Void, Void> {
private static final String TAG = "TestAPITask2";
private String myAccessToken;
private String myRefreshToken;
@Override
protected Void doInBackground(Void... voids) {
MyApiRetrofit loginService = ServiceAuthGenerator.getTokenService(MyApiRetrofit.class, "client_id", "client_secret");
Call<Authorize> call = loginService.accessToken("password", "myusername", "mypassword");
call.enqueue(new Callback<Authorize>() {
@Override
public void onResponse(Call<Authorize> call, Response<Authorize> response) {
if (response.isSuccessful()) {
// user object available
Log.i(TAG, "isSuccessful");
Authorize temp = response.body();
myAccessToken = temp.getAccessToken();
Log.i(TAG, temp.toString());
MyApiRetrofit myInfoService = ServiceInfoGenerator.retrieveInfoService(MyApiRetrofit.class, myAccessToken, "myusername");
Call<Authorize> call2 = myInfoService.retrieveMyInfo();
try {
Authorize user = call2.execute().body();
} catch (IOException e) {
e.printStackTrace();
}
} else {
// error response, no access to resource?
Log.i(TAG, "error response");
}
}
@Override
public void onFailure(Call<Authorize> call, Throwable t) {
// something went completely south (like no internet connection)
Log.d(TAG, t.getMessage());
}
});
return null;
}
}
我按照here的说明操作。 我的改造版本是:
compile 'com.squareup.retrofit2:retrofit:2.0.2'
compile 'com.squareup.retrofit2:converter-gson:2.0.2'
compile 'com.squareup.okhttp3:logging-interceptor:3.2.0'
上次记录:
10-15 13:17:41.434 9968-9968 D/OkHttp: --> POST https://oauth.reddit.com/api/v1/me http/1.1
10-15 13:17:41.434 9968-9968 D/OkHttp: Content-Length: 0
10-15 13:17:41.434 9968-9968 D/OkHttp: Authorization: bearer tokenToken
10-15 13:17:41.434 9968-9968 D/OkHttp: User-Agent: MyRedditClient/0.1 by myusername
10-15 13:17:41.434 9968-9968 D/OkHttp: --> END POST (0-byte body)
10-15 13:17:41.988 9968-9968 D/OkHttp: <-- 404 Not Found https://oauth.reddit.com/api/v1/me (553ms)
10-15 13:17:41.988 9968-9968 D/OkHttp: Content-Type: application/json; charset=UTF-8
10-15 13:17:41.988 9968-9968 D/OkHttp: x-frame-options: SAMEORIGIN
10-15 13:17:41.988 9968-9968 D/OkHttp: x-content-type-options: nosniff
10-15 13:17:41.988 9968-9968 D/OkHttp: x-xss-protection: 1; mode=block
10-15 13:17:41.989 9968-9968 D/OkHttp: expires: -1
10-15 13:17:41.989 9968-9968 D/OkHttp: cache-control: private, s-maxage=0, max-age=0, must-revalidate, max-age=0, must-revalidate
10-15 13:17:41.990 9968-9968 D/OkHttp: x-ratelimit-remaining: 598.0
10-15 13:17:41.990 9968-9968 D/OkHttp: x-ratelimit-used: 2
10-15 13:17:41.990 9968-9968 D/OkHttp: x-ratelimit-reset: 136
10-15 13:17:41.990 9968-9968 D/OkHttp: set-cookie: loid=Fsx2GnGYmufCQZ6cfT; Domain=reddit.com; Max-Age=63071999; Path=/; expires=Mon, 15-Oct-2018 10:17:44 GMT; secure
10-15 13:17:41.991 9968-9968 D/OkHttp: set-cookie: loidcreated=2016-10-15T10%3A17%3A44.173Z; Domain=reddit.com; Max-Age=63071999; Path=/; expires=Mon, 15-Oct-2018 10:17:44 GMT; secure
10-15 13:17:41.991 9968-9968 D/OkHttp: x-ua-compatible: IE=edge
10-15 13:17:41.991 9968-9968 D/OkHttp: set-cookie: loid=Q52c9gouzuGGdg7UXW; Domain=reddit.com; Max-Age=63071999; Path=/; expires=Mon, 15-Oct-2018 10:17:44 GMT; secure
10-15 13:17:41.991 9968-9968 D/OkHttp: set-cookie: loidcreated=2016-10-15T10%3A17%3A44.189Z; Domain=reddit.com; Max-Age=63071999; Path=/; expires=Mon, 15-Oct-2018 10:17:44 GMT; secure
10-15 13:17:41.991 9968-9968 D/OkHttp: X-Moose: majestic
10-15 13:17:41.991 9968-9968 D/OkHttp: Content-Length: 38
10-15 13:17:41.992 9968-9968 D/OkHttp: Accept-Ranges: bytes
10-15 13:17:41.992 9968-9968 D/OkHttp: Date: Sat, 15 Oct 2016 10:17:44 GMT
10-15 13:17:41.992 9968-9968 D/OkHttp: Via: 1.1 varnish
10-15 13:17:41.992 9968-9968 D/OkHttp: Connection: keep-alive
10-15 13:17:41.992 9968-9968 D/OkHttp: X-Served-By: cache-ams4426-AMS
10-15 13:17:41.992 9968-9968 D/OkHttp: X-Cache: MISS
10-15 13:17:41.992 9968-9968 D/OkHttp: X-Cache-Hits: 0
10-15 13:17:41.992 9968-9968 D/OkHttp: X-Timer: S1476526664.126987,VS0,VE109
10-15 13:17:41.993 9968-9968 D/OkHttp: Server: snooserv
10-15 13:17:42.002 9968-9968 D/OkHttp: {"message": "Not Found", "error": 404}
10-15 13:17:42.002 9968-9968 D/OkHttp: <-- END HTTP (38-byte body)
答案 0 :(得分:1)
您对端点/api/v1/me
的API调用是POST
方法
@POST("/api/v1/me")
Call<Authorize> retrieveMyInfo();
根据他们http://www-stud.rbi.informatik.uni-frankfurt.de/~markus/antiword/的点/api/v1/me
,您应该使用GET
次请求。
@GET("/api/v1/me")
Call<Authorize> retrieveMyInfo();
我没有测试过该解决方案,但它似乎是404的原因,documentation它解决了问题(对于python)。
答案 1 :(得分:0)
可能的原因之一可能是您的BASE URL。
根据Retrofit 2,基本网址不得以&#39; /&#39;结尾。
请删除&#39; /&#39;从基本URL并尝试一个。
在Retrofit 1.9中,它运行良好。
干杯!!!
答案 2 :(得分:0)
由于Retrofit 2.0如何使用Http Resolve来解析端点的uri方案,如果你将baseurl指定为此http://hello.com
,将端点URL指定为/ world / foo,它将会中断。
您需要使用基本网址http://hello.com/
和端点网址world / foo。
/有所作为。
如果有帮助请通知我,否则我会给你另一个解决方案。 您的问题与我认为的问题相同:Link
您可以查看此问题:Link
答案 3 :(得分:0)
@POST("api/v1/me")
在第一个字符网址中删除“/”,因为改造2.0需要BASEURL必须以“/”结尾。我在代码中找不到其他问题。如果有任何空值,您应该检查您的标头,或者检查API链接
答案 4 :(得分:0)
我终于找到了解决方案!我用错误的grant_type打电话!我改变它,现在它正在工作!