Проблема с настройкой заголовка "Content-Type" при загрузке файла с HttpClient4
Я пытаюсь загрузить файл (или несколько файлов) в мой сервлет, который использует Apache file-upload для обработки и получения файлов после публикации.
Все идет хорошо, и файл отправляется и принимается, когда я использую следующий код.
DefaultHttpClient client = new DefaultHttpClient();
HttpPost post = new HttpPost("http://myservice.com/servlet");
MultipartEntity entity2 = new MultipartEntity();
FileBody fileBody = new FileBody(new File("C:/docOut.pdf"));
entity2.addPart("file", fileBody);
post.setEntity(entity2);
HttpResponse httpResponse = client.execute(post);
System.out.println(EntityUtils.toString(httpResponse.getEntity()));
Но когда я пытаюсь установить свой собственный "Content-Type" на рекомендованный (или тот, который принимается только в библиотеке загрузки файлов Apache) с загрузкой файла:
post.addHeader("Content-Type", "multipart/form-data");
Мой сервлет не получает ни одного файла и выдает исключение:
org.apache.commons.fileupload.FileUploadException: the request was rejected because no multipart boundary was found
at org.apache.commons.fileupload.FileUploadBase$FileItemIteratorImpl.<init>(FileUploadBase.java:931)
at org.apache.commons.fileupload.FileUploadBase.getItemIterator(FileUploadBase.java:331)
at org.apache.commons.fileupload.FileUploadBase.parseRequest(FileUploadBase.java:349)
at org.apache.commons.fileupload.servlet.ServletFileUpload.parseRequest(ServletFileUpload.java:126)
at com.myservice.server.filerep.action.FileUploadFormAction.execute(FileUploadFormAction.java:54)
at com.myservice.server.filerep.web.FileRepServlet.doGet(FileRepServlet.java:34)
at com.myservice.server.filerep.web.FileRepServlet.doPost(FileRepServlet.java:41)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:637)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
at org.mortbay.jetty.servlet.ServletHolder.handle(ServletHolder.java:487)
at org.mortbay.jetty.servlet.ServletHandler.handle(ServletHandler.java:362)
at org.mortbay.jetty.security.SecurityHandler.handle(SecurityHandler.java:216)
at org.mortbay.jetty.servlet.SessionHandler.handle(SessionHandler.java:181)
at org.mortbay.jetty.handler.ContextHandler.handle(ContextHandler.java:729)
at org.mortbay.jetty.webapp.WebAppContext.handle(WebAppContext.java:405)
at org.mortbay.jetty.handler.HandlerWrapper.handle(HandlerWrapper.java:152)
at org.mortbay.jetty.handler.RequestLogHandler.handle(RequestLogHandler.java:49)
at org.mortbay.jetty.handler.HandlerWrapper.handle(HandlerWrapper.java:152)
at org.mortbay.jetty.Server.handle(Server.java:324)
at org.mortbay.jetty.HttpConnection.handleRequest(HttpConnection.java:505)
at org.mortbay.jetty.HttpConnection$RequestHandler.content(HttpConnection.java:843)
at org.mortbay.jetty.HttpParser.parseNext(HttpParser.java:647)
at org.mortbay.jetty.HttpParser.parseAvailable(HttpParser.java:211)
at org.mortbay.jetty.HttpConnection.handle(HttpConnection.java:380)
at org.mortbay.io.nio.SelectChannelEndPoint.run(SelectChannelEndPoint.java:395)
at org.mortbay.thread.QueuedThreadPool$PoolThread.run(QueuedThreadPool.java:488)
Я знаю, что POST-запросы с загружаемым файлом не должны содержать граничные "элементы" в запросе, чтобы определить порядок загруженных кусков байтов, но я думал, что HttpClient добавит всю необходимую информацию в мой запрос (аналогично тому, как когда я не указывать тип контента).
Мои вопросы:
Почему добавление "Content-Type" нарушило мой запрос? Разве HttpClient не должен добавлять граничные элементы к определенному мной типу контента?
Должен ли я установить "Content-Type" в моем запросе подробно или позволить библиотеке обработать его?
Если я могу установить
Content-Type
Объясните, можете ли вы предоставить фрагмент кода?Если я могу явно установить Content-Type, почему я должен использовать "multipart/form-data", а не "application/x-www-form-urlencoded", когда дело касается POSTing для некоторых форм?
PS: я нашел как-то связанные вопросы, но не решая мою проблему:
Проблема ContentType с клиентом commons-upload и httpcomponent
2 ответа
Если у вас есть тип данных enctype, вы должны следовать правилам, указанным в RFC 2388. Данные в составном сообщении рассматриваются как объект, поэтому каждый объект должен иметь заголовок (с Content-Disposition
, Content-Type
и т. д.) и границы.
Отвечая на вопрос 1, RFC заявляет:
Как и во всех составных MIME-типах, каждая часть имеет необязательный "Content-Type", который по умолчанию равен text/plain.
Для 2), как уже упоминалось, каждое составное сообщение должно иметь заголовок, поэтому вы должны указать свой Content-Type (если вы не используете функции библиотеки HttpClient полностью).
Для 3) и 4) RFC заявляет:
Если несколько файлов должны быть возвращены в результате одной записи формы, они должны быть представлены в виде части "multipart/mixed", встроенной в "multipart/form-data".
Надеюсь это поможет.
Для fedd/ класс, который вставляет случайно сгенерированную границу, является HttpClient, а не HttpPost. так что вы должны посмотреть на методы для HttpClient