Is it possible to download a file with HTTP POST?

2020-02-26 03:54发布

问题:

Is it possible to download a file with HTTP POST? I know the "Get" way(windows.location), but in my case, there are a lot of param that should be passed to server

回答1:

Yes, the rest of a POST request can direct a browser to download a file. The file contents would be sent as the HTTP response, same as in the GET case.



回答2:

You mean like this ?

    function IssuePostRequest(objData) 
    {
        var strPageURL = "about:blank";
        var strAction = "@Url.Action("GetPDF", "Home")/";
        //var strAction = "/popups/delete.aspx";

        var strWindowName = "MyEvilHttpPostInAnewWindow"; // ifrmDownload
        var iWindowWidth = 805;
        var iWindowHeight = 625;



        var form = document.createElement("form");
        form.setAttribute("id", "bla");
        form.setAttribute("method", "post");
        form.setAttribute("action", strAction);
        form.setAttribute("target", strWindowName);
        form.setAttribute("style", "display: none;");
        // setting form target to a window named 'formresult'


        // Repeat for all data fields
        var hiddenField = document.createElement("input");
        hiddenField.setAttribute("name", "data");
        hiddenField.setAttribute("value", objData);
        form.appendChild(hiddenField);
        // End Repeat for all data fields


        document.body.appendChild(form);



        // creating the 'formresult' window with custom features prior to submitting the form
        //window.open(test.html, 'formresult', 'scrollbars=no,menubar=no,height=600,width=800,resizable=yes,toolbar=no,status=no');
        //JS_PopupCenterScreen(strPageURL, strWindowName, iWindowWidth, iWindowHeight);
        window.open(strPageURL, strWindowName);

        // document.forms[0].submit();
        //document.getElementById("xxx").click();
        form.submit();
    } // End Function IssuePostRequest

With this Server code:

    public FileResult GetPDF(string data)
    {
        //data = @"";

        string base64Data = System.Text.RegularExpressions.Regex.Match(data, @"data:image/(?<type>.+?),(?<data>.+)").Groups["data"].Value;
        byte[] binData = Convert.FromBase64String(base64Data);

        byte[] ba = PdfHandler.ImageToPdf(binData);
        //System.IO.File.WriteAllBytes(@"d:\temp\myba.pdf", ba);

        //return System.Convert.ToBase64String(ba);
        return File(ba, "application/pdf", "Chart.pdf");
    }


回答3:

Looks like you'd like to generate the POST request from Javascript. I believe there is no way to get the browser to treat the result of an AJAX request as a download. Even if the Content-Type is set to something that browsers would normally offer as a download (e.g. to "application/octet-stream"), the browser will only deposit the data in the XMLHttpRequest object.

Furthermore, as you probably already know, there is no way to make window.open() issue a POST request.

I think the best way is to make an AJAX request which generates a file on the server. On the browser, when that request completes, use window.open() to download the generated file.



回答4:

In some sense, every HTTP GET or POST is "downloading a file", but it's better to think of it as the message payload rather than a file. In most cases, the payload is an HTML document that the browser should render as a web page. But what if it's not an HTML document? What if it's a zip file for which the browser should offer the user a "Save as" dialog? Obviously, the browser must make a determination about the content type of the response and handle it correctly.

One of the most common ways that a browser determines the content type is through a HTTP header called, accordingly, "Content-Type". This header takes the value of a mime-type. This is the key to browsers doing content specific things like firing up an acrobat plugin when the response contains a pdf file, etc.

Note, not all browsers 1) determine the content type in the same way, and 2) react to the content type in the same way. Sometimes you have to toy with setting the headers to get the behaviors you want from all the browsers. All server side technologies allow you to set HTTP headers.



回答5:

There's no difference, other than the request method and how you send data to the server. The way you process the response is the same regardless of whether you use GET or POST.



回答6:

I managed to solve it using this:

service.js

downloadExcel : function() {
    var mapForm = document.createElement("form");
    mapForm.target ="_self"||"_blank";
    mapForm.id="stmtForm";
    mapForm.method = "POST";
    mapForm.action = "your_Controller_URL";

    var mapInput = document.createElement("input");
    mapInput.type = "hidden";
    mapInput.name = "Data";
    mapForm.appendChild(mapInput);
    document.body.appendChild(mapForm);

    mapForm.submit();
}

Spring Controller Code :

@Controller

@PostMapping(value = "/your_Controller_URL")
    public void doDownloadEmsTemplate( final HttpServletRequest request, final HttpServletResponse response)
            throws IOException, URISyntaxException {

        String filePath = "/location/zzzz.xls";
        logger.info("Excel Template File Location Path :" + filePath);
        final int BUFFER_SIZE = 4096;
        ServletContext context = request.getServletContext();
        String appPath = context.getRealPath("");
        String fullPath = appPath + filePath;
        File downloadFile = new File(fullPath);
        FileInputStream inputStream = new FileInputStream(downloadFile);
        String mimeType = context.getMimeType(fullPath);
        if (mimeType == null) {
            //mimeType = "application/octet-stream";
            mimeType = "application/vnd.ms-excel";
        }
        logger.info("MIME type: " + mimeType);
        response.setContentType(mimeType);
        response.setContentLength((int) downloadFile.length());
        String headerKey = "Content-Disposition";
        String headerValue = String.format("attachment; filename=\"%s\"", downloadFile.getName());
        logger.info("File Download Successfully : ");
        response.setHeader(headerKey, headerValue);
        OutputStream outStream = response.getOutputStream();
        byte[] buffer = new byte[BUFFER_SIZE];
        int bytesRead = -1;
        while ((bytesRead = inputStream.read(buffer)) != -1) {
            outStream.write(buffer, 0, bytesRead);
        }
        inputStream.close();
        outStream.close();
    }