Reputation: 2242
I need help.
I have an endpoint that takes a parameter. Depending on this parameter, the JSON
returned will be completely different.
Is it possible for RetroFit
to handle this?
For example:
http://myserver.com/all/<parameter>
where parameter is BUS
or BICYCLE
, as well as adding others later.
An example BUS
request will return:
"stops": [{
"Lat": "....",
"Lng": "....",
"Name": "Main Street",
"Route": "",
"StopNumber": "2"
}]
The BICYCLE
endpoint will return:
"stops": [{
"address": "Town Centre",
"lat": "....",
"lng": "....",
"number": "63",
"open": "1"
}]
Depending on which section of my Android app that the user is in, I would like to send a different parameter, and be able to handle it using the same call.
I was thinking of using a parent class called AllTypes
that each of the other classes extend, and then setting my Retrofit
call signature to:
@GET("/all/{type}")
void getAll(@Path("type") String type, Callback<AllTypes> callback);
But I'm not sure if Retrofit
can automatically pick the correct child class for AllTypes
based on the returned JSON
, or even the passed parameter type
.
Does anyone know how to do this? If not, I'll just have to create multiple different methods with different Callback types.
Thanks.
Upvotes: 11
Views: 16003
Reputation: 11
I know it's a bit late, but probably my tip will be useful for someone.
If a returned JSON content depends on an endpoint parameter "type"(for example) one can use PolymorphicJsonAdapterFactory from Moshi for decoding to certain class. Please note the returned JSON must contain type information.
Upvotes: 1
Reputation: 1934
you can use JsonElement in such as follow : in ApiInterface.java :
@GET("web api url")
Call<JsonElement> GetAllMyClass();
than in MyActivity :
ApiInterface client = ApiClient.getClient().create(ApiInterface.class);
Call<JsonElement> call = client.GetAllMyClass();
call.enqueue(new Callback<JsonElement>() {
@Override
public void onResponse(Call<JsonElement> call, Response<JsonElement> response) {
if (response.isSuccessful()) {
JsonElement questions = response.body();
// if response type is array
List<MyClass> response_array = new Gson().fromJson(((JsonArray)questions), new TypeToken<List<MyClass>>(){}.getType());
// if response type is object
MyClass response_one = new Gson().fromJson(questions.getAsJsonObject(), MyClass.class);
} else {
Log.d("MyClassCallback", "Code: " + response.code() + " Message: " + response.message());
}
}
@Override
public void onFailure(Call<JsonElement> call, Throwable t) {
t.printStackTrace();
}
});
Upvotes: 3
Reputation: 91
In this response, since it is only change in the fields but not the structure like array or object. In that case just define all the fields in the java object, the values that will be returned in that specific call will be mapped and the others will be null. Put a logic on your end to check null checks.
Upvotes: 0
Reputation: 2242
Just to close this off.
You can get the raw JSON back from RetroFit
and use GSON
to manually serialise to the class type you want.
For example:
RestClient.get().getDataFromServer(params, new Callback<JSONObject>() {
@Override
public void success(JSONObject json, Response response) {
if(isTypeA) {
//Use GSON to serialise to TypeA
} else {
//Use GSON to serialise to TypeB
}
}
});
Upvotes: 12