Модернизация нескольких типов ответов - PullRequest
1 голос
/ 28 января 2020

Как я могу использовать Retrofit2 для анализа этих двух типов ответов API?

Ok ответ (HTTP 200):

{
    "data": {
        "foo": "bar"
    }
}

Ошибка ответа (HTTP 200):

{
    "error": {
        "foo": "bar"
    }
}

Я прочитал тонны SO вопросов и руководств, но я не знаю, как это сделать, я попробовал:

GsonBuilder gsonBuilder = new GsonBuilder();
gsonBuilder.registerTypeAdapterFactory(new ItemTypeAdapterFactory());
Gson gson = gsonBuilder.create();

final Retrofit retrofit = new Retrofit.Builder()
        .client(getOkHttpClient())
        .baseUrl(Constants.API_BASE_URL)
        .addConverterFactory(GsonConverterFactory.create(gson))
        .build();

И это мой ItemTypeAdapterFactory:

class ItemTypeAdapterFactory implements TypeAdapterFactory {

    public <T> TypeAdapter<T> create(final Gson gson, final TypeToken<T> type) {

        final TypeAdapter<T> delegate = gson.getDelegateAdapter(this, type);
        final TypeAdapter<JsonElement> elementAdapter = gson.getAdapter(JsonElement.class);

        return new TypeAdapter<T>() {

            public void write(JsonWriter out, T value) throws IOException {
                delegate.write(out, value);
            }

            public T read(JsonReader in) throws IOException {

                JsonElement jsonElement = elementAdapter.read(in);

                if (jsonElement.isJsonObject()) {
                    JsonObject jsonObject = jsonElement.getAsJsonObject();

                    // Data key
                    if (jsonObject.has(Constants.JSON_KEY_DATA)) {

                        JsonElement jsonData = jsonObject.get(Constants.JSON_KEY_DATA);

                        // Primitive
                        if (jsonData.isJsonPrimitive()) {
                            jsonElement = jsonData.getAsJsonPrimitive();
                        }
                        // JSON object
                        else if (jsonData.isJsonObject()) {
                            jsonElement = jsonData;
                        }
                        // JSON object array
                        else if (jsonData.isJsonArray()) {
                            jsonElement = jsonData.getAsJsonArray();
                        }
                    }
                }

                return delegate.fromJsonTree(jsonElement);
            }
        }.nullSafe();
    }
}

Но теперь я не знаю тип, который будет объявлен на интерфейсе retrofit2, внутри Call:

@GET("login")
Call<?> login(@Query(Constants.API_PARAM_TOKEN) String token);

Не могли бы вы указать мне правильное направление?

1 Ответ

1 голос
/ 28 января 2020

В аналогичном случае я однажды использовал JsonObject в качестве типа, поэтому ваша функция будет выглядеть следующим образом:

@GET("login")
Call<?> login(@Query(Constants.API_PARAM_TOKEN) String token);

Далее, когда вы делаете модифицированный вызов, вы сохраняете ответ в виде строки , Итак, в вашем java коде сделайте что-то вроде этого:

Call<JsonObject> call = RetrofitClient.getAPIService().login('YOUR_INPUT');
Data data = null;
Error error = null;
call.enqueue(new Callback<JsonObject>() {
            @Override
            public void onResponse(Call<JsonObject> call, Response<JsonObject> response) {
if(response.isSuccessfull()){
  String jsonString = response.body().toString();
  if(jsonString.contains("data:")){
       data = new Gson().fromJson(jsonString,Data.class);
  }else{
      error = new Gson().fromJson(jsonString,Error.class);
  }
}
        }

Здесь я использовал Data и Error эти 2 класса. Они POJO. Таким образом, данные могут выглядеть примерно так:

Data.java:

public class Data implements Serializable{
 @SerializedName("foo")
    @Expose
    private Foo foo; // Foo is your desired data type 

}

То же самое относится и к Error. Поэтому, в зависимости от вашего кода, внесите необходимые изменения. Удачи.

...