2011-09-27 14 views
15

È possibile scaricare un file con HTTP POST? Conosco il modo "Get" (windows.location), ma nel mio caso ci sono molti parametri che dovrebbero essere passati al serverÈ possibile scaricare un file con HTTP POST?

+0

Questo è il vantaggio del POST, è possibile inviare molti dati (p.es. payload). La sicurezza è anche lì, ma nel tuo caso l'invio di parametri è il requisito. – EMM

risposta

14

Sì, il resto di una richiesta POST può indirizzare un browser per scaricare un file. Il contenuto del file verrebbe inviato come risposta HTTP, come nel caso GET.

3

Non c'è differenza, oltre al metodo di richiesta e alla modalità di invio dei dati al server. Il modo in cui elabori la risposta è lo stesso indipendentemente dal fatto che tu usi GET o POST.

+0

Ciao Greg, puoi fare un esempio?Grazie mille – Sean

4

In un certo senso, ogni HTTP GET o POST sta "scaricando un file", ma è meglio pensarlo come il carico utile del messaggio piuttosto che un file. Nella maggior parte dei casi, il payload è un documento HTML che il browser dovrebbe visualizzare come una pagina web. Ma cosa succede se non è un documento HTML? Cosa succede se si tratta di un file zip per il quale il browser dovrebbe offrire all'utente una finestra di dialogo "Salva come"? Ovviamente, il browser deve stabilire il tipo di contenuto della risposta e gestirlo correttamente.

Uno dei modi più comuni con cui un browser determina il tipo di contenuto è tramite un HTTP header chiamato, di conseguenza, "Content-Type". Questa intestazione prende il valore di un tipo mime. Questa è la chiave per i browser che eseguono contenuti specifici come l'attivazione di un plug-in di acrobat quando la risposta contiene un file pdf, ecc.

Nota, non tutti i browser 1) determinano il tipo di contenuto nello stesso modo e 2) reagiscono al tipo di contenuto nello stesso modo. A volte devi giocare con l'impostazione delle intestazioni per ottenere i comportamenti che desideri da tutti i browser. Tutte le tecnologie lato server consentono di impostare le intestazioni HTTP.

12

Sembra che si desideri generare la richiesta POST da Javascript. Credo che non ci sia modo di ottenere che il browser tratti il ​​risultato di una richiesta AJAX come download. Anche se Content-Type è impostato su qualcosa che i browser normalmente offrono come download (ad esempio "application/octet-stream"), il browser depositerà i dati solo nell'oggetto XMLHttpRequest.

Inoltre, come probabilmente già sapete, non c'è modo di far apparire window.open() una richiesta POST.

Penso che il modo migliore sia creare una richiesta AJAX che generi un file sul server. Nel browser, al termine della richiesta, utilizzare window.open() per scaricare il file generato.

+0

Questa è una delle possibili soluzioni, l'altra sta facendo l'invio del modulo per recuperare il file appropriato. –

11

Intendi questo?

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 

Con questo codice server:

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"); 
    } 
+0

Questo è davvero utile, grazie. L'unico problema è che apre una nuova finestra. Potrebbe essere aperto in un iFrame? –

+2

@Menelaos Vergis: aggiungi un alla tua pagina. –

+0

@Quandry: Grazie mille, non so perché questa risposta non abbia ottenuto l'attenzione che merita, funziona come previsto. –

0

sono riuscito a risolverlo utilizzando questo:

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

Codice controller Primavera:

@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(); 
    } 
Problemi correlati