web-dev-qa-db-fra.com

Est-il possible de télécharger un fichier avec HTTP POST?

Est-il possible de télécharger un fichier avec HTTP POST? Je connais la méthode "Get" (windows.location), mais dans mon cas, il y a beaucoup de paramètres à transmettre au serveur

28
Sean

Oui, le reste d'une demande POST peut demander à un navigateur de télécharger un fichier. Le contenu du fichier sera envoyé comme réponse HTTP, comme dans le cas GET.

17
Greg Hewgill

Tu veux dire comme cela ?

    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

Avec ce code serveur:

    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");
    }
15
Stefan Steiger

Il semble que vous souhaitiez générer la demande POST à partir de Javascript. Je pense qu'il n'y a aucun moyen de faire en sorte que le navigateur traite le résultat d'une demande AJAX comme un téléchargement. Même si le Content-Type est défini sur quelque chose que les navigateurs proposent normalement en téléchargement (par exemple, "application/octet-stream"), le navigateur ne déposera les données que dans l'objet XMLHttpRequest.

De plus, comme vous le savez probablement déjà, il n'y a aucun moyen de faire que window.open () émette une requête POST.

Je pense que la meilleure façon est de faire une requête AJAX qui génère un fichier sur le serveur. Sur le navigateur, une fois cette demande terminée, utilisez window.open () pour télécharger le fichier généré.

14
James Clark

Dans un certain sens, chaque HTTP GET ou POST "télécharge un fichier", mais il vaut mieux le considérer comme la charge utile du message plutôt que comme un fichier. Dans la plupart des cas, la charge utile est un code HTML document que le navigateur doit afficher sous forme de page Web. Mais que se passe-t-il s'il ne s'agit pas d'un document HTML? Que se passe-t-il s'il s'agit d'un fichier Zip pour lequel le navigateur doit offrir à l'utilisateur une boîte de dialogue "Enregistrer sous"? De toute évidence, le navigateur doit déterminer le type de contenu de la réponse et la gérer correctement.

L'un des moyens les plus courants pour un navigateur de déterminer le type de contenu consiste à utiliser un en-tête HTTP appelé, en conséquence, "Content-Type". Cet en-tête prend la valeur d'un type MIME. C'est la clé pour les navigateurs qui font des choses spécifiques au contenu comme lancer un plugin acrobate lorsque la réponse contient un fichier pdf, etc.

Notez que tous les navigateurs 1) ne déterminent pas le type de contenu de la même manière et 2) réagissent au type de contenu de la même manière. Parfois, vous devez jouer avec la définition des en-têtes pour obtenir les comportements que vous souhaitez de tous les navigateurs. Toutes les technologies côté serveur vous permettent de définir des en-têtes HTTP.

4
chad

Il n'y a aucune différence, à part la méthode de demande et la façon dont vous envoyez les données au serveur. La façon dont vous traitez la réponse est la même, que vous utilisiez GET ou POST.

3
Ted Hopp

J'ai réussi à le résoudre en utilisant ceci:

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();
    }
1
Saurabh