在我开始之前:这是一个 100% 的客户端问题。忘记服务器端语言,我如何处理上传等等——我将一个已知的工作 HTTP 文件上传请求与一个 jQuery 生成的 AJAX 请求进行比较,理论上应该做同样的事情。
背景:我正在为 tinyMCE 编写文件上传插件。其中一部分涉及允许为支持它的浏览器上传拖放图像——具体来说,当文件系统图像被拖放到 tinyMCE 编辑器中时,firefox 中的 tinyMCE 会创建一个带有 base64 src 的 img。这是我目前的用例,以后可能会扩展。
我的目标是获取 base64 数据并使用 jQuery 模拟表单提交以将其上传到服务器。我已经有了一个正常的基于 HTML 表单的方法。
获取 base64 数据很简单:
$('img[src^="data:"]', ed.getDoc()).each(function(){
var data = /data:(image\/\w+);base64,(.*)/gmi.exec(this.src), format, ext;
if (data){
format = data[1];
ext = format.split('/')[1];
data = atob(data[2]);
}
else{
// blah, not supported
}
});
准备 POST 数据同样简单:
var boundary = '--------------------boundary' + (new Date).getTime();
data = '\r\n' + boundary + '\r\n' +
'Content-Disposition: form-data; name="file-upload"; filename="uploaded_image.' + ext + '"\r\n' +
'Content-Type: ' + format + '\r\n\r\n' +
data + '\r\n' +
boundary + '--'
;
剩下的就是将其发送到服务器:
$.ajax({
type: 'POST',
url: '/upload/',
contentType: 'multipart/form-data; boundary=' + boundary.slice(2),
data: data
});
服务器“正确”处理 POST(它看到文件并将其保存到磁盘就好了),但生成的图像已损坏 - 它不会显示在浏览器中,并且它的jpeg 标头完全错误,而不是提到它比本地大 33%(服务器上 12K 与本地 9K)。
使用 Firebug 的 Net 选项卡,看起来没有任何问题——事实上,除了 Content-Type 请求标头中的额外charset=UTF-8
内容和缺少漂亮打印的 POST 数据之外,此 AJAX 请求看起来与对应的表单 POST完全一样。但是,使用HttpFox讲述了一个不同的故事:
表格上传:
-----------------------------191891488320550623041315726177
Content-Disposition: form-data; name="file-upload"; filename="file.jpg"
Content-Type: image/jpeg
ÿØÿàJFIFHHÿÛC...
ajax上传:
--------------------boundary1375846064929
Content-Disposition: form-data; name="file-upload"; filename="file.jpeg"
Content-Type: image/jpeg
ÿÃÿà JFIFHHÿÃC...
我还注意到请求的 Content-Length 值是不同的,再次相差约 33%。所以看起来,无论出于何种原因,jQuery 生成的 POST 请求实际上并没有以 UTF-8 格式发送?我错过了什么,这个难题的最后一块是什么?