Как обрабатывать многостраничный запрос, состоящий из файла и объекта JSON, в Spring услуге восстановления?

У меня есть следующий ресурс (реализованный с использованием Spring 4.05.RELEASE), который принимает файл и объект JSON:

(P.S. activityTemplate - сериализуемый класс сущностей)

...
@RequestMapping(value="/create", method=RequestMethod.POST)
public @ResponseBody ActivityTemplate createActivityTemplate(
        @RequestPart ActivityTemplate activityTemplate, @RequestPart MultipartFile jarFile)
{
   //process the file and JSON
}
...

и это форма, которую я тестирую:

<form method="POST" enctype="multipart/form-data"
    action="http://localhost:8080/activityTemplates/create">
    JSON: <input type="text" name="activityTemplate" value='/* the JSON object*/'><br />

    File to upload: <input type="file" name="file">
    <input type="submit" value="Upload">
</form>

и это ошибка, которую я получаю:

 There was an unexpected error (type=Unsupported Media Type, status=415).
 Content type 'application/octet-stream' not supported

Итак, как мне заставить ресурс принять объект JSON как часть многостраничного запроса или мне нужно отправить форму другим способом?

Ответы

Ответ 1

Мне потребовалось два дня, чтобы работать на меня!

клиент (angular):

$scope.saveForm = function () {
      var formData = new FormData();
      var file = $scope.myFile;
      var json = $scope.myJson;
      formData.append("file", file);
      formData.append("ad",JSON.stringify(json));//important: convert to string JSON!
      var req = {
        url: '/upload',
        method: 'POST',
        headers: {'Content-Type': undefined},
        data: formData,
        transformRequest: function (data, headersGetterFunction) {
          return data;
        }
      };

Spring (Boot):

@RequestMapping(value = "/upload", method = RequestMethod.POST)
    public @ResponseBody
    Advertisement storeAd(@RequestPart("ad") String adString, @RequestPart("file") MultipartFile file) throws IOException {

        Advertisement jsonAd = new ObjectMapper().readValue(adString, Advertisement.class);
//do whatever you want with your file and jsonAd

Ответ 2

Надеюсь, это поможет вам. Вам необходимо установить границу в запросе, чтобы сообщить HTTP-запрос.  просто; Краткое введение в многостраничный формат можно найти в приведенной ниже ссылке

HTML 4.01 Спецификация для multipart

Следующий пример иллюстрирует кодировку multipart/form-data​​strong > . Если Json Object - " MyJsonObj", а файл, который необходимо отправить, " myfile.txt", пользовательский агент может отправить обратно следующие данные:

Content-Type: multipart/form-data; boundary=MyBoundary

--MyBoundary
Content-Disposition: form-data; name="myJsonString"
Content-Type: application/json

MyJsonObj //Your Json Object goes here
--MyBoundary
Content-Disposition: form-data; name="files"; filename="myfile.txt"
Content-Type: text/plain

... contents of myfile.txt ...
--MyBoundary--

или если ваши файлы имеют тип изображения с именем " image.gif", затем

--MyBoundary
Content-Disposition: file; filename="image.gif"
Content-Type: image/gif
Content-Transfer-Encoding: binary

...contents of image.gif...
--MyBoundary--

Вы указываете boundary в Content-Type header, чтобы сервер знал, как разделить отправленные данные.

Итак, вам в основном нужно выбрать граничное значение для:

  • Используйте значение, которое не будет отображаться в HTTP-данных, отправленных на сервер, например 'AaB03x'.
  • Будьте единообразны и используйте одно и то же значение по всему запросу.

Ответ 3

Вы не указали имена параметров на @RequestPart s?

public @ResponseBody ActivityTemplate createActivityTemplate(
    @RequestPart("activityTemplate") ActivityTemplate activityTemplate, @RequestPart("file") MultipartFile jarFile)
{
   //process the file and JSON
}

Примечание. Не забудьте включить в jQuery файл jQuer (файл Json to ActivityTemplate) в вашем пути к классам.

Ответ 4

Не могли бы вы изменить свой

@RequestMapping(value="/create", method=RequestMethod.POST)

к

@RequestMapping(value="/create",
                method=RequestMethod.POST, consumes ={"multipart/form-data"})

Ответ 5

Тип содержимого по умолчанию - "application/octet-stream". Поскольку вы загружаете jar файл и JSON, тип содержимого должен быть указан в аннотации @RequestMapping следующим образом:

@RequestMapping(value="/create", method=RequestMethod.POST, headers="content-type=application/json,application/java-archive")

Ответ 6

Сообщение об ошибке указывает, что HttpMessageConverter не зарегистрирован для части контента с несколькими частями /MIME: application/octet-stream. Тем не менее, ваш параметр jarFile, скорее всего, правильно идентифицируется как application/octet-stream, поэтому я предполагаю наличие несоответствия в сопоставлении параметров.

Итак, сначала попробуйте установить одно и то же имя для параметра и элемента ввода формы.

Другая проблема заключается в том, что JSON загружается как (регулярное) значение текстового ввода в форме, а не как отдельная часть в multi-part/MIME. Поэтому нет заголовка содержимого, связанного с ним, чтобы узнать, что Spring должен использовать десериализатор JSON. Вместо этого вы можете использовать @RequestParam и зарегистрировать конкретный конвертер, как в этом ответе: параметр JSON в Spring MVC-контроллере

Ответ 7

это может помочь вам, получая MultipartFile, вы должны установить тип содержимого заголовка запроса на "multipart/form-data", а затем в вашем контроллере использовать consumes = "multipart/form-data", потребляет также используемый для сопоставления нашего запроса с нашим методом в контроллере.

Если вы хотите получать данные JSON, лучше отправить запрос в форме JSONString, просто получите этот jsonstring, позже конвертируйте в формат json Object, затем используйте этот объект для своих операций.

проверьте ниже код:

@RequestMapping(value="/savingImg", method=RequestMethod.POST, 
        consumes="multipart/form-data", produces="application/json")
public ResponseEntity<?> savingAppIMgDtlss(
        @RequestParam(value="f1", required = false) MultipartFile f1 , 
        @RequestParam(value="f2", required = false) MultipartFile f2 ,
        @RequestParam(value="f3", required = false) MultipartFile f3 ,
        @RequestParam(value="f4", required = false) MultipartFile f4 ,
        @RequestParam(value="f5", required = false) MultipartFile f5 ,
        @RequestParam(value="f6", required = false) MultipartFile f6 ,
        @RequestParam(value="f7", required = false) MultipartFile f7 ,
        @RequestParam(value="f8", required = false) MultipartFile f8 ,@RequestParam("data") String jsonString) 
                throws Exception , ParseException {
    try{
        JSONObject gstcTranObj = new JSONObject();
                //converting JSONString to JSON
        net.sf.json.JSONObject jsonDtls = net.sf.json.JSONObject.fromObject(jsonString);
        System.out.println("f1::"+f1.getOriginalFilename());
        System.out.println("f2::"+f2.getOriginalFilename());
        System.out.println("f3::"+f3.getOriginalFilename());
        System.out.println("f4::"+f4.getOriginalFilename());
        System.out.println("f5::"+f5.getOriginalFilename());
        System.out.println("f6::"+f6.getOriginalFilename());
        System.out.println("f7::"+f7.getOriginalFilename());
        System.out.println("f8::"+f8.getOriginalFilename());
} catch (Exception e) {
        e.printStackTrace();

        return new ResponseEntity<>("Failed",HttpStatus.NOT_FOUND);
    }finally{

    }
return new ResponseEntity<>("Success", HttpStatus.OK);

  }
}

Ответ 8

Возникает исключение, потому что у вас нет соответствующего HttpMessageConverter для обработки запроса multipart/form-data. Временное решение

Ответ 9

Вы можете использовать @RequestPart из org.springframework.web.bind.annotation.RequestPart; Он используется как сочетание @RequestBody и загрузки файла.

Используя @RequestParam как этот @RequestParam ("файл") MultipartFile файл, вы можете загрузить только файл и несколько отдельных данных (значение ключа), как

        @RequestMapping(value = "/uploadFile", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
    public void saveFile(
                         @RequestParam("userid") String userid,
                         @RequestParam("file") MultipartFile file) {

    }

вы можете публиковать данные объекта JSON и файл File, используя @RequestPart, как

    @RequestMapping(value = "/patientp", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
public ResponseEntity<?> insertPatientInfo(
                                            @RequestPart PatientInfoDTO patientInfoDTO,
                                            @RequestPart("file") MultipartFile file) {
}

Вы не ограничены использованием загрузки нескольких файлов непосредственно в качестве параметров метода контроллера. Ваши объекты формы могут содержать поля Part или MultipartFile, и Spring автоматически знает, что он должен получить значения из файловых частей, и соответствующим образом преобразует их.

Вышеуказанный метод может отвечать на ранее продемонстрированный многочастный запрос, содержащий один файл. Это работает, потому что Spring имеет встроенный конвертер сообщений HTTP, который распознает части файла. В дополнение к типу javax.servlet.http.Part вы также можете преобразовать загружаемые файлы в org.springframework.web.multipart.MultipartFile. Если поле файла разрешает многократную загрузку файлов, как показано во втором многочастном запросе, просто используйте массив или набор частей или MultipartFiles.

        @RequestMapping(value = "/patientp", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
    public ResponseEntity<?> insertPatientInfo(
                                                @RequestPart PatientInfoDTO patientInfoDTO,
                                                @RequestPart("files") List<MultipartFile> files) {
    }

Рад помочь...