дооснащения имеет Converter.Factory
абстрактный класс, который можно использовать для выполнять пользовательское представление HTTP. Вы можете создать конвертер для построения okhttp.RequestBody
, если метод имеет конкретную аннотацию.
Конечный результат будет выглядеть следующим образом:
@POST("/")
Call<Void> postBar(@Body @Root("bar") String foo)
и преобразование: postBar("Hello World")
в { "bar" : "Hello World" }
.
Давайте начнем.
Шаг 1 - создать аннотацию для корневого ключа (Root.java)
/**
* Denotes the root key of a JSON request.
* <p>
* Simple Example:
* <pre><code>
* @POST("/")
* Call<ResponseBody> example(
* @Root("name") String yourName);
* </code></pre>
* Calling with {@code foo.example("Bob")} yields a request body of
* <code>{name=>"Bob"}</code>.
* @see JSONConverterFactory
*/
@Documented
@Target(PARAMETER)
@Retention(RUNTIME)
public @interface Root {
/**
* The value of the JSON root.
* Results in {"value" : object}
*/
String value();
}
Шаг 2 - определите Converter.Factory, который определяет аннотацию (JSONConverterFactory.java). Я использую Gson для разбора JSON, но вы можете использовать любую инфраструктуру, которую хотите.
/**
* Converts @Root("key") Value to {"key":json value} using the provided Gson converter.
*/
class JSONConverterFactory extends Converter.Factory {
private final Gson gson;
private static final MediaType CONTENT_TYPE =
MediaType.parse("application/json");
JSONConverterFactory(Gson gson) {
this.gson = gson;
}
@Override
public Converter<?, RequestBody> requestBodyConverter(
Type type, Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {
for (Annotation annotation : parameterAnnotations) {
if (annotation instanceof Root) {
Root rootAnnotation = (Root) annotation;
return new JSONRootConverter<>(gson, rootAnnotation.value());
}
}
return null;
}
private final class JSONRootConverter<T> implements Converter<T, RequestBody> {
private Gson gson;
private String rootKey;
private JSONRootConverter(Gson gson, String rootKey) {
this.gson = gson;
this.rootKey = rootKey;
}
@Override
public RequestBody convert(T value) throws IOException {
JsonElement element = gson.toJsonTree(value);
JsonObject object = new JsonObject();
object.add(this.rootKey, element);
return RequestBody.create(CONTENT_TYPE, this.gson.toJson(object));
}
}
}
Шаг 3 - установить JSONConverterFactory в свой Retrofit например
Gson gson = new GsonBuilder().create(); // Or your customized version
Retrofit.Builder builder = ...;
builder.addConverterFactory(new JSONConverterFactory(gson))
Шаг 4 - Прибыль
@POST("/")
Call<Void> postBar(@Body @Root("bar") String foo)
Или для Вашего случая:
@POST("foo/{fooId}/bars")
Observable<Void> postBar(@Body @Root("bar") String barValue, @Path("fooId") String styleId);
Смотрите также это [мета-обсуждение] (http://meta.stackoverflow.com/q/327487/230513). – trashgod