下载.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类型。
答案 0 :(得分:1)
此帖子末尾的GET请求仅提取文件内容。它不会获取文件。因此,文件内容被发回可能缺少MS文件的必要编码步骤,从而抛出(此时,完全合法)解析器错误。
这是一个不稳定的解决方案,但看起来端点可以就内容和mime类型达成一致,以便将文件呈现给最终用户。
遗憾的是,我们无法控制最终的GET请求,只能将文件数据作为BLOB访问,并且相信mime类型的编码是正确的。