下载XLSX文件

时间:2015-08-17 17:45:26

标签: jquery ajax mime-types

下载.xlsx文件时,请求成功但我的ajax错误回调会抛出“parsererror - Invalid xml”消息。 我的应用程序从服务器请求一个文件,该服务器是从另一个服务器获取该文件的代理。

在所有三台机器上,我在.htaccess中设置了以下内容:

AddType application/vnd.openxmlformats-officedocument.spreadsheetml.sheet xlsx

ajax调用的结构是这样的:

var attachment = { name="myfile.xlsx", 
                   content_type="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet xlsx",  
                   id="resource_id_on_remote-server"}

var path = '/path_to/attachment_file.php?' + $.param(attachment);

$.ajax({
  type: "GET",
  url: document.getElementById('attachment_download_frame').src = path,
  success: function(xhrResponse) {
  },
  error: function(xhrResponse, errmsg, error) {
    //errmsg = "parsererror"
  },
  complete: function() {
    $('#spinner').spin("modal");
  }
});

我通过隐藏的Iframe汇集了这个电话,该iframe会启动请求。

服务器文件attachment_file.php使用“附件”json参数从主机服务器请求附件文件:

$id = filter_input(INPUT_GET, 'id');
  $contentType = filter_input(INPUT_GET, 'content_type');
  $name = filter_input(INPUT_GET, 'name');

  $url = REMOTE_BASE_URL . '/api/v1/attachment.json?';
  $url .= http_build_query(array('id' => $id, 'name' => $name, 'content_type' => $contentType));

  $curl = curl_init($url);
  $headers = array(
      'Content-Type: ' . $contentType
  );

  curl_setopt($curl, CURLOPT_HTTPHEADER, $headers);
  curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
  curl_setopt($curl, CURLOPT_TIMEOUT, 30);
  curl_setopt($curl, CURLOPT_HTTPGET, true);

  $response = curl_exec($curl);
  $httpCode = curl_getinfo($curl, CURLINFO_HTTP_CODE);

  curl_close($curl);
  if ($response && in_array($httpCode, array(200, 201, 204))) {
    header("Cache-Control: must-revalidate, post-check=0, pre-check=0");
    header('Content-Description: File Transfer');
    header("Content-Type: {$contentType}");
    header("Content-Disposition: attachment; filename={$name}");
    header("Expires: 0");
    header("Pragma: public");
    $fh = @fopen('php://output', 'w');
    fputs($fh, $response);
  } else {

  }

请注意如何保留“Content-Type”。我已经确认这确实设置为:“application / vnd.openxmlformats-officedocument.spreadsheetml.sheet xlsx”

调用的api服务器是Rails应用程序,使用传入的参数构建文件资源请求:

  def fetch_attachment options
    url = REMOTE_URL + '/my_attachments/' + options[:id]
    response = HTTParty.get(url)
    if status? response
      return response.parsed_response
    else
      return false;
    end
    return false;
  end

使用HTTParty.get作为请求。在facepalm的这个时刻,我将调查HTTParty请求中的内容类型是否丢失。

此外,我出于特定原因从参数设置mime类型。

1 个答案:

答案 0 :(得分:1)

此帖子末尾的GET请求仅提取文件内容。它不会获取文件。因此,文件内容被发回可能缺少MS文件的必要编码步骤,从而抛出(此时,完全合法)解析器错误。

这是一个不稳定的解决方案,但看起来端点可以就内容和mime类型达成一致,以便将文件呈现给最终用户。

遗憾的是,我们无法控制最终的GET请求,只能将文件数据作为BLOB访问,并且相信mime类型的编码是正确的。