2011-09-27 16 views
15

¿Es posible descargar un archivo con HTTP POST? Conozco el camino "Get" (windows.location), pero en mi caso, hay una gran cantidad de parámetro que debe ser transmitida al servidor¿Es posible descargar un archivo con HTTP POST?

+0

Esa es la ventaja de POST, puede enviar muchos datos (también conocidos como carga útil). La seguridad también está ahí, pero en su caso el envío de parámetros es el requisito. – EMM

Respuesta

14

Sí, el resto de una solicitud POST puede dirigir un navegador para descargar una archivo. El contenido del archivo se enviará como la respuesta HTTP, igual que en el caso GET.

3

No hay diferencia, aparte del método de solicitud y la forma de enviar datos al servidor. La forma en que procesa la respuesta es la misma independientemente de si usa GET o POST.

+0

Hola Greg, ¿puedes dar un ejemplo?Muchas gracias – Sean

4

En cierto sentido, cada HTTP GET o POST es "descargar un archivo", pero es mejor considerarlo como la carga útil del mensaje en lugar de un archivo. En la mayoría de los casos, la carga útil es un documento HTML que el navegador debe representar como una página web. Pero, ¿y si no es un documento HTML? ¿Qué sucede si se trata de un archivo comprimido para el cual el navegador debe ofrecer al usuario un cuadro de diálogo "Guardar como"? Obviamente, el navegador debe tomar una determinación sobre el tipo de contenido de la respuesta y manejarlo correctamente.

Una de las formas más comunes en que un navegador determina el tipo de contenido es a través de un HTTP header llamado, en consecuencia, "Content-Type". Este encabezado toma el valor de un tipo mime. Esta es la clave para los navegadores que realizan contenido específico, como encender un complemento acrobat cuando la respuesta contiene un archivo pdf, etc.

Nota: no todos los navegadores 1) determinan el tipo de contenido de la misma manera, y 2) reaccionan al tipo de contenido de la misma manera. A veces hay que jugar con configurar los encabezados para obtener los comportamientos que desea de todos los navegadores. Todas las tecnologías del lado del servidor le permiten establecer encabezados HTTP.

12

Parece que desea generar la solicitud POST de Javascript. Creo que no hay forma de que el navegador trate el resultado de una solicitud de AJAX como descarga. Incluso si Content-Type se configura como algo que los navegadores normalmente ofrecerían como descarga (por ejemplo, "application/octet-stream"), el navegador solo depositará los datos en el objeto XMLHttpRequest.

Además, como probablemente ya sepa, no hay forma de que window.open() emita una solicitud POST.

Creo que la mejor manera es hacer una solicitud AJAX que genere un archivo en el servidor. En el navegador, cuando se complete esa solicitud, use window.open() para descargar el archivo generado.

+0

Esta es una de las soluciones posibles, la otra está haciendo la presentación del formulario para recuperar el archivo apropiado. –

11

¿Te refieres a esto?

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 este código de servidor:

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

Esto es realmente útil, gracias. El único problema es que abre una nueva ventana. ¿Podría esto abrirse en un iFrame? –

+2

@Menelaos Vergis: Agregue un a su página. –

+0

@Quandry: Muchas gracias, no sé por qué esta respuesta no ha recibido la atención que merece, funciona como se esperaba. –

0

que lograron resolverlo utilizando la siguiente:

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

código del controlador de 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(); 
    }