Форма Resttemplate/составная: изображение + JSON в POST

Я пытаюсь вызвать rest ws (используя resttemplate), который принимает изображение и некоторый JSON. Тем не менее, я не могу заставить его работать.

Соответствующий код выглядит следующим образом:

    HttpHeaders header = new HttpHeaders();
    header.setContentType(MediaType.MULTIPART_FORM_DATA);

    MultiValueMap<String, Object> map = new LinkedMultiValueMap<>();
    ByteArrayResource bytes = new ByteArrayResource(pictureData) {
        @Override
        public String getFilename() {
            return pictureName;
        }
    };
    map.add("x", x);
    map.add("file", bytes);

    HttpEntity<MultiValueMap<String, Object>> requestEntity = new HttpEntity(map, header);
    String response =  restTemplate.postForObject(UPLOAD_URL, requestEntity, String.class);

Где x — это какой-то POJO со всеми необходимыми аннотациями JSON (я получаю его от другого веб-сервиса, эта часть работает нормально).

Эта штука, однако, говорит мне: HttpMessageNotWritableException: Не удалось написать запрос: не найдено подходящего HttpMessageConverter для x.

Если я изменю ByteArrayResource на byte[], то получу ошибку 400 Bad Request. Если я изменю тип контента на JSON, ByteArrayResource не сможет быть сериализован в JSON:

Caused by: org.springframework.http.converter.HttpMessageNotWritableException: Could not write JSON: No serializer found for class java.io.ByteArrayInputStream and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) ) (through reference chain: org.springframework.util.LinkedMultiValueMap["file"]->java.util.LinkedList[0]->a.b.c.["inputStream"]); nested exception is com.fasterxml.jackson.databind.JsonMappingException: No serializer found for class java.io.ByteArrayInputStream and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) ) (through reference chain: org.springframework.util.LinkedMultiValueMap["file"]->java.util.LinkedList[0]->a.b.c.["inputStream"])

У меня настроены следующие преобразователи:

StringHttpMessageConverter,
MappingJackson2HttpMessageConverter
FormHttpMessageConverter

Есть идеи, пожалуйста? Заранее спасибо.

ОБНОВЛЕНИЕ

Итак, что у меня на данный момент после инструкции: я регистрирую преобразователи так:

    RestTemplate restTemplate = new RestTemplate();
    restTemplate.getMessageConverters().add(new StringHttpMessageConverter());
    restTemplate.getMessageConverters().add(new MappingJackson2HttpMessageConverter());

    FormHttpMessageConverter formHttpMessageConverter = new FormHttpMessageConverter();
        formHttpMessageConverter.addPartConverter(new MappingJackson2HttpMessageConverter());
        formHttpMessageConverter.addPartConverter(new ResourceHttpMessageConverter()); // This is hope driven programming

    restTemplate.getMessageConverters().add(new ResourceHttpMessageConverter());
    restTemplate.getMessageConverters().add(formHttpMessageConverter);

Затем в вызове ws у меня есть:

    HttpHeaders header = new HttpHeaders();
    header.setContentType(MediaType.APPLICATION_JSON); //Also tried with multipart...

    MultiValueMap<String, Object> multipartRequest = new LinkedMultiValueMap<>();

    ByteArrayResource bytes = new ByteArrayResource(pictureData) {
        @Override
        public String getFilename() {
            return pictureName;
        }
    };

    HttpHeaders xHeader = new HttpHeaders();
    xHeader.setContentType(MediaType.APPLICATION_JSON);
    HttpEntity<X> xPart = new HttpEntity<>(x, xHeader);
    multipartRequest.add("x", xPart);

    HttpHeaders pictureHeader = new HttpHeaders();
    pictureHeader.setContentType(MediaType.IMAGE_PNG);
    HttpEntity<ByteArrayResource> picturePart = new HttpEntity<>(bytes, pictureHeader);
    multipartRequest.add("file", picturePart);

    HttpEntity<MultiValueMap<String, Object>> requestEntity = new HttpEntity(multipartRequest, header);
    return restTemplate.postForObject(UPLOAD_URL, requestEntity, String.class);

person TamasGyorfi    schedule 21.03.2015    source источник


Ответы (1)


Если вы хотите использовать ByteArrayResource, просто зарегистрируйте ResourceHttpMessageConverter.

Если вы хотите использовать byte[], просто зарегистрируйте ByteArrayHttpMessageConverter.

Тип содержимого части изображения должен быть типом изображения, например image/png, а не application/json.

Вы можете установить тип данных каждой отдельной части с помощью

HttpHeaders partHeaders = new HttpHeaders();
partHeaders.setContentType(MediaType.IMAGE_PNG);
HttpEntity<ByteArrayResource> bytesPart = new HttpEntity<ByteArrayResource>(bytes, partHeaders);

map.add("file", bytesPart);

Создайте свой RestTemplate, предоставив свою коллекцию HttpMessageConverter

HttpMessageConverter<Object> jackson = new MappingJackson2HttpMessageConverter();
HttpMessageConverter<Resource> resource = new ResourceHttpMessageConverter();
FormHttpMessageConverter formHttpMessageConverter = new FormHttpMessageConverter();
formHttpMessageConverter.addPartConverter(jackson);
formHttpMessageConverter.addPartConverter(resource); // This is hope driven programming

RestTemplate restTemplate = new RestTemplate(Arrays.asList(jackson, resource, formHttpMessageConverter));

и ваш внешний HttpEntity должен иметь составной тип контента

header.setContentType(MediaType.MULTIPART_FORM_DATA);
person Sotirios Delimanolis    schedule 21.03.2015