restTemplate Spring-mvc와 함께 멀티파트 폼 데이터를 전송하는 방법
저는 Raspberry Pi에 RestTemplate 파일을 Jetty와 함께 업로드하려고 합니다.파이에는 다음과 같은 서블릿이 실행되고 있습니다.
protected void doPost(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
PrintWriter outp = resp.getWriter();
StringBuffer buff = new StringBuffer();
File file1 = (File) req.getAttribute("userfile1");
String p = req.getParameter("path");
boolean success = false;
if (file1 == null || !file1.exists()) {
buff.append("File does not exist\n");
} else if (file1.isDirectory()) {
buff.append("File is a directory\n");
} else {
File outputFile = new File(req.getParameter("userfile1"));
if(isValidPath(p)){
p = DRIVE_ROOT + p;
final File finalDest = new File(p
+ outputFile.getName());
success = false;
try {
copyFileUsingFileChannels(file1, finalDest);
finalDest.setWritable(true);
success = true;
} catch (Exception e) {
e.printStackTrace();
}
if (success){
buff.append("File successfully uploaded.\n");
}
else{
buff.append("Failed to save file.");
}
}
else{
buff.append("Invalid path.\n");
}
}
outp.write(buff.toString());
}
컬로 성공적으로 할 수 있습니다.
curl --form userfile1=@/home/pi/src/CreateNewFolderServlet.java --form press=OK localhost:2222/pi/GetFileServlet?path="/media/"
이것은 웹앱에서도 동일한 기능을 가져야 하는 방법입니다.
@ResponseBody
@RequestMapping(value="/upload/",method=RequestMethod.POST ,produces = "text/plain")
public String uploadFile(MultipartHttpServletRequest request2, HttpServletResponse response2){
Iterator<String> itr = request2.getFileNames();
MultipartFile file = request2.getFile(itr.next());
System.out.println(file.getOriginalFilename() +" uploaded!");
System.out.println(file.toString());
MultiValueMap<String, Object> parts = new LinkedMultiValueMap<String, Object>();
parts.add("userfile1",file);
//reqEntity.addPart("userfile1", file);
String path="/public/";
RestTemplate restTemplate = new RestTemplate();
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.MULTIPART_FORM_DATA);
System.out.println("1");
HttpEntity<MultiValueMap<String, Object>> request = new HttpEntity<MultiValueMap<String, Object>>(parts, headers);
String url = url2+"/pi/GetFileServlet?path="+path;
System.out.println("2");
/* restTemplate.getMessageConverters().add(new FormHttpMessageConverter());
restTemplate.getMessageConverters().add(
new MappingJackson2HttpMessageConverter());*/
System.out.println("3");
ResponseEntity<String> response = restTemplate.exchange(url, HttpMethod.POST, request,String.class);
System.out.println("4");
System.out.println("response : " +response);
if(response==null||response.getBody().trim()==""){
return "error";
}
return response.getBody();
}
결과는 다음과 같습니다.
ui-https.whttps 업로드!
organ.springframework.web.webart.support.Standard MultipartHttpServletRequest$Standard MultipartFile@47e7673e
1
2
3
보시다시피 4번은 인쇄되지 않았습니다 콘솔에서 예외는 없습니다.디버깅 중에 발견된 예외:
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.web.multipart.support.StandardMultipartFile["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.web.multipart.support.StandardMultipartFile["inputStream"])
파일 전체를 읽어 들이는 중입니다.ByteArrayResource
대용량 파일의 경우 메모리 소모 문제가 될 수 있습니다.
스프링 mvc 컨트롤러에서 파일 업로드를 프록시할 수 있습니다.InputStreamResource
:
@RequestMapping(value = "/upload", method = RequestMethod.POST)
public ResponseEntity<?> uploadImages(@RequestPart("images") final MultipartFile[] files) throws IOException {
LinkedMultiValueMap<String, Object> map = new LinkedMultiValueMap<>();
String response;
HttpStatus httpStatus = HttpStatus.CREATED;
try {
for (MultipartFile file : files) {
if (!file.isEmpty()) {
map.add("images", new MultipartInputStreamFileResource(file.getInputStream(), file.getOriginalFilename()));
}
}
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.MULTIPART_FORM_DATA);
String url = "http://example.com/upload";
HttpEntity<LinkedMultiValueMap<String, Object>> requestEntity = new HttpEntity<>(map, headers);
response = restTemplate.postForObject(url, requestEntity, String.class);
} catch (HttpStatusCodeException e) {
httpStatus = HttpStatus.valueOf(e.getStatusCode().value());
response = e.getResponseBodyAsString();
} catch (Exception e) {
httpStatus = HttpStatus.INTERNAL_SERVER_ERROR;
response = e.getMessage();
}
return new ResponseEntity<>(response, httpStatus);
}
class MultipartInputStreamFileResource extends InputStreamResource {
private final String filename;
MultipartInputStreamFileResource(InputStream inputStream, String filename) {
super(inputStream);
this.filename = filename;
}
@Override
public String getFilename() {
return this.filename;
}
@Override
public long contentLength() throws IOException {
return -1; // we do not want to generally read the whole stream into memory ...
}
}
RestTemplate의 기본 MessageConverters 중 MultipartFile 파일에 포함된 InputStream을 직렬화하는 방법을 아는 사람이 없으므로 예외가 발생합니다.RestTemplate를 통해 객체를 보낼 때 대부분 POJO를 보내길 원합니다.MultiPartFile 자체 대신 MultiValueMap에 MultiPartFile의 바이트를 추가하면 이 문제를 해결할 수 있습니다.
당신의 서블릿 부분도 문제가 있는 것 같습니다.예를 들어.
File file1 = (File) req.getAttribute("userfile1");
ServletRequest의 getAttribute 메서드는 요청/양식 매개 변수를 반환하지 않고 Servlet 컨텍스트에서 설정한 특성을 반환하므로 항상 null을 반환해야 합니다.당신의 컬 예제에서 실제로 작동하는 것이 확실합니까?
다음은 파일을 서블릿에 전달하는 Spring MVC 메서드의 예입니다.
서블릿(Spring MVC 컨테이너에서 실행하는 테스트를 했음에도 불구하고)은 여기서 채택되었습니다.
@RequestMapping("/pi")
private void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
final String path = request.getParameter("destination");
final Part filePart = request.getPart("file");
final String fileName = request.getParameter("filename");
OutputStream out = null;
InputStream fileContent = null;
final PrintWriter writer = response.getWriter();
try {
out = new FileOutputStream(new File(path + File.separator
+ fileName));
fileContent = filePart.getInputStream();
int read = 0;
final byte[] bytes = new byte[1024];
while ((read = fileContent.read(bytes)) != -1) {
out.write(bytes, 0, read);
}
writer.println("New file " + fileName + " created at " + path);
} catch (FileNotFoundException fne) {
writer.println("You either did not specify a file to upload or are "
+ "trying to upload a file to a protected or nonexistent "
+ "location.");
writer.println("<br/> ERROR: " + fne.getMessage());
} finally {
if (out != null) {
out.close();
}
if (fileContent != null) {
fileContent.close();
}
if (writer != null) {
writer.close();
}
}
}
스프링 MVC 방식:
@ResponseBody
@RequestMapping(value="/upload/", method=RequestMethod.POST,
produces = "text/plain")
public String uploadFile(MultipartHttpServletRequest request)
throws IOException {
Iterator<String> itr = request.getFileNames();
MultipartFile file = request.getFile(itr.next());
MultiValueMap<String, Object> parts =
new LinkedMultiValueMap<String, Object>();
parts.add("file", new ByteArrayResource(file.getBytes()));
parts.add("filename", file.getOriginalFilename());
RestTemplate restTemplate = new RestTemplate();
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.MULTIPART_FORM_DATA);
HttpEntity<MultiValueMap<String, Object>> requestEntity =
new HttpEntity<MultiValueMap<String, Object>>(parts, headers);
// file upload path on destination server
parts.add("destination", "./");
ResponseEntity<String> response =
restTemplate.exchange("http://localhost:8080/pi",
HttpMethod.POST, requestEntity, String.class);
if (response != null && !response.getBody().trim().equals("")) {
return response.getBody();
}
return "error";
}
이 방법을 사용하면 다음과 같은 컬을 사용하여 MVC 방식으로 파일을 성공적으로 서블릿에 업로드할 수 있습니다.
curl --form file=@test.dat localhost:8080/upload/
버전 5.1 이후 Spring Framework는 자체적으로 제공됩니다.Resource
을 위한 실행MultipartFile
s. 따라서 로렌조의 대답은 다음을 제거함으로써 단순화할 수 있습니다.MultipartInputStreamFileResource
다음과 같이 클래스와 맵 채우기:
[...]
for (MultipartFile file : files) {
if (!file.isEmpty()) {
map.add("images", file.getResource());
}
}
[...]
언급URL : https://stackoverflow.com/questions/28408271/how-to-send-multipart-form-data-with-resttemplate-spring-mvc
'programing' 카테고리의 다른 글
jQuery를 사용하여 div에서 id 속성을 제거하는 방법? (0) | 2023.09.15 |
---|---|
regex 및 Powershell이 있는 문자열에서 특정 데이터 추출 (0) | 2023.09.15 |
하나의 SELECT 문에 두 개의 스칼라 변수를 설정하시겠습니까? (0) | 2023.09.15 |
PHP 객체 리터럴 (0) | 2023.09.15 |
기판을 사용하여 Oracle에서 문자열 자르기 (0) | 2023.09.15 |