2011-09-27 42 views
15

Có thể tải xuống tệp bằng HTTP POST không? Tôi biết cách "Nhận" (windows.location), nhưng trong trường hợp của tôi, có rất nhiều thông số cần được chuyển đến máy chủCó thể tải xuống tệp bằng HTTP POST không?

+0

Đó là lợi thế của POST, bạn có thể gửi nhiều dữ liệu (còn gọi là tải trọng). An ninh cũng có nhưng trong trường hợp của bạn gửi các tham số là yêu cầu. – EMM

Trả lời

14

Có, phần còn lại của yêu cầu POST có thể hướng trình duyệt tải xuống tập tin. Nội dung tệp sẽ được gửi dưới dạng phản hồi HTTP, giống như trong trường hợp GET.

3

Không có sự khác biệt, ngoài phương thức yêu cầu và cách bạn gửi dữ liệu đến máy chủ. Cách bạn xử lý phản hồi là như nhau bất kể bạn sử dụng GET hay POST.

+0

Xin chào Greg, bạn có thể đưa ra một ví dụ không?Cảm ơn rất nhiều – Sean

4

Trong một số ý nghĩa, mỗi HTTP GET hoặc POST là "tải xuống tệp", nhưng tốt hơn nên nghĩ đó là tải trọng thư chứ không phải là tệp. Trong hầu hết các trường hợp, tải trọng là một tài liệu HTML mà trình duyệt sẽ hiển thị dưới dạng trang web. Nhưng nếu nó không phải là một tài liệu HTML thì sao? Điều gì xảy ra nếu đó là tệp zip mà trình duyệt sẽ cung cấp cho người dùng hộp thoại "Lưu dưới dạng"? Rõ ràng, trình duyệt phải đưa ra quyết định về loại nội dung của phản hồi và xử lý chính xác.

Một trong những cách phổ biến nhất mà trình duyệt xác định loại nội dung là thông qua số HTTP header được gọi, theo đó, "Loại nội dung". Tiêu đề này lấy giá trị của một loại mime. Đây là chìa khóa cho các trình duyệt thực hiện những nội dung cụ thể như kích hoạt plugin acrobat khi phản hồi chứa tệp pdf, v.v.

Lưu ý, không phải tất cả trình duyệt 1) xác định loại nội dung theo cùng một cách và 2) phản hồi đối với loại nội dung theo cùng một cách. Đôi khi bạn phải chơi với thiết lập các tiêu đề để có được những hành vi bạn muốn từ tất cả các trình duyệt. Tất cả các công nghệ phía máy chủ đều cho phép bạn đặt tiêu đề HTTP.

12

Có vẻ như bạn muốn tạo yêu cầu POST từ Javascript. Tôi tin rằng không có cách nào để trình duyệt xử lý kết quả của một yêu cầu AJAX dưới dạng tải xuống. Ngay cả khi Content-Type được đặt thành thứ mà trình duyệt thường cung cấp dưới dạng bản tải xuống (ví dụ: "ứng dụng/octet-stream"), trình duyệt sẽ chỉ gửi dữ liệu trong đối tượng XMLHttpRequest.

Hơn nữa, như bạn có thể đã biết, không có cách nào để tạo window.open() đưa ra yêu cầu POST.

Tôi nghĩ cách tốt nhất là thực hiện yêu cầu AJAX tạo tệp trên máy chủ. Trên trình duyệt, khi yêu cầu đó hoàn tất, hãy sử dụng window.open() để tải xuống tệp đã tạo.

+0

Đây là một trong những giải pháp khả thi, cách khác là gửi biểu mẫu để truy xuất tệp thích hợp. –

11

Ý bạn là như thế này?

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 

Với mã này 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

Điều này thực sự hữu ích, cảm ơn bạn. Vấn đề duy nhất là nó mở một cửa sổ mới. Điều này có thể được mở trong một iFrame không? –

+2

@Menelaos Vergis: Thêm vào trang của bạn. –

+0

@Quandry: Cảm ơn bạn rất nhiều, tôi không biết tại sao câu trả lời này không nhận được sự chú ý xứng đáng, nó hoạt động như mong đợi. –

0

tôi quản lý để giải quyết nó bằng cách sử này:

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

Xuân điều khiển Mã:

@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(); 
    } 
Các vấn đề liên quan