ServletFileUpload#parseRequest(request) returns an empty list

26,335

Solution 1

It will be empty at that point if you have already (implicitly) parsed the request body beforehand. The HTTP request body can be read/parsed only once (as the client has sent it only once and won't send it multiple times).

The request body will be implicitly read/parsed when you have invoked any of the following methods before feeding the request to Commons FileUpload:

request.getParameter();
request.getParameterMap();
request.getParameterNames();
request.getParameterValues();
request.getReader();
request.getInputStream();

You need to make absolutely sure that you are not calling any of those methods beforehand (also check all servlet filters to be sure).

If you've already ensured that you aren't doing that, then the only other possible causes would be an incorrect boundary header and/or using incorrect newlines (it has really to be CR+LF and thus not alone LF). You can find a concrete and proper example at the bottom of Using java.net.URLConnection to fire and handle HTTP requests, under the section "Uploading files".

Solution 2

If spring is used, check if there is a <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver" /> in xml file. There is a function

protected HttpServletRequest checkMultipart(HttpServletRequest request) throws MultipartException {
        if (this.multipartResolver != null && this.multipartResolver.isMultipart(request)) {
            if (request instanceof MultipartHttpServletRequest) {
                logger.debug("Request is already a MultipartHttpServletRequest - if not in a forward, " +
                        "this typically results from an additional MultipartFilter in web.xml");
            }
            else {
                return this.multipartResolver.resolveMultipart(request);
            }
        }
        // If not returned before: return original request.
        return request;
    }

"eats" the multipart items. It causes a three hours pain.

Share:
26,335
toto
Author by

toto

Programmer: Java, JavaScript, Jquery, Android, PHP, HTML5, .NET, NodeJS Contact: Email: [email protected]

Updated on May 17, 2020

Comments

  • toto
    toto about 4 years

    I have a client application in Android which uses HttpURLConnection to send files to the server. The server uses the Apache Commons FileUpload API to parse the form-data values.

    The HttpURLConnection sends this request:

    -----------------------------4912995119421
    Content-Disposition: form-data; name="deviceid"
    
    9428103
    -----------------------------4912995119421
    Content-Disposition: form-data; name="countryid"
    
    598
    -----------------------------4912995119421
    Content-Disposition: form-data; name="number"
    
    98621360
    -----------------------------4912995119421
    Content-Disposition: form-data; name="file"; filename="2012-12-08 17.42.18.jpg"
    Content-Type: image/jpeg
    
    ÿØÿá1 Exif  II*    
          @      °      ª    
       ²              ¼       Ä   (       1    Ì   2    Ø          i‡    ì   %ˆ    \  n  SAMSUNG GT-S5360L H      H      S5360LUHLB1 2012:12:08 17:42:18  š‚    î  ?‚    ö  "ˆ       'ˆ    È    ?    0220?    þ  ?      ‘     ’    &  ’       
    ’    .        0100             @       °       >  £       ¤        ¤        ¤    6  ¤                     
       2012:12:08 17:42:18 2012:12:08 17:42:18    
         d               R98      0100                         (           ¤      T.      ÿØÿà JFIF      ÿÛ C @@ÿÛ 
    -----------------------------4912995119421--
    

    The server code:

        String contentType = request.getContentType();
        if ((contentType.indexOf("multipart/form-data") == -1)) {
            response.setStatus(HttpServletResponse.SC_NOT_FOUND);
            return;
        }
    
    
        long maxFileSize = (2 * 1024 * 1024);
    
        int maxMemSize = (2 * 1024 * 1024);
    
        DiskFileItemFactory factory = new DiskFileItemFactory();
        // maximum size that will be stored in memory
        factory.setSizeThreshold(maxMemSize);
    
        // Create a new file upload handler
        ServletFileUpload upload = new ServletFileUpload(factory);
        // maximum file size to be uploaded.
    
        upload.setSizeMax(maxFileSize);
    
        List fileItems = upload.parseRequest(request);
        Iterator i = fileItems.iterator();
    
    
        //leo primero todas las variables.
    
        int deviceID = 0;
        int countryID = 0;
        String phoneNumber = "";
    
    
    
        while (i.hasNext()) {
            FileItem fi = (FileItem) i.next();
    
            if (fi.isFormField()) {
                String variable = fi.getFieldName();
                if (variable.equals("deviceid")) {
                    deviceID = Integer.parseInt(fi.getString());
                } else if (variable.equals("countryid")) {
                    countryID = Integer.parseInt(fi.getString());
                } else if (variable.equals("number")) {
                    phoneNumber = String.valueOf(Long.parseLong(fi.getString()));
                }
    
            }
    
        }
    
        if (deviceID == 0 || countryID == 0 || phoneNumber.equals("")) {
            response.setStatus(HttpServletResponse.SC_NOT_FOUND);
            return;
        }
    

    The problem is in the line List fileItems = upload.parseRequest(request);. The returned list is empty and I can not get the form-data values.