Как обрабатывать многостраничный запрос, состоящий из файла и объекта 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-datastrong > .
Если 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) {
}
Рад помочь...