出于教育目的,我想创建文件块上传。当你们上传所有的块时,你们怎么知道?
我尝试从temp
移动块并重命名它们以使它们的顺序正确,然后将最后一个块合并在一起。然而,最后一件作品不是收到的最后一件,我猜。所以块上的fopen()
失败了,因为它们尚未创建,我得到的最终文件的大小与上一个块的大小完全相同。
我相信我可以使用.onload
xhr
上的function upload(file) {
var BYTES_PER_CHUNK = parseInt(2097152, 10),
size = file.size,
NUM_CHUNKS = Math.max(Math.ceil(SIZE / BYTES_PER_CHUNK), 1),
start = 0, end = BYTES_PER_CHUNK, num = 1;
var chunkUpload = function(blob) {
var fd = new FormData();
var xhr = new XMLHttpRequest();
fd.append('upload', blob, file.name);
fd.append('num', num);
fd.append('num_chunks', NUM_CHUNKS);
xhr.open('POST', '/somedir/upload.php', true);
xhr.send(fd);
}
while (start < size) {
chunkUpload(file.slice(start, end));
start = end;
end = start + BYTES_PER_CHUNK;
num++;
}
}
事件逐个发送数据块,这样我就不必将它们从PHP临时移动,但我想知道如果有不同的解决方案。
取悦你的一些基本代码:
$target_path = ROOT.'/upload/';
$tmp_name = $_FILES['upload']['tmp_name'];
$filename = $_FILES['upload']['name'];
$target_file = $target_path.$filename;
$num = $_POST['num'];
$num_chunks = $_POST['num_chunks'];
move_uploaded_file($tmp_name, $target_file.$num);
if ($num === $num_chunks) {
for ($i = 1; $i <= $num_chunks; $i++) {
$file = fopen($target_file.$i, 'rb');
$buff = fread($file, 2097152);
fclose($file);
$final = fopen($target_file, 'ab');
$write = fwrite($final, $buff);
fclose($final);
unlink($target_file.$i);
}
}
和PHP:
aSelectedDataSet.forEach(function(currentObject,index){
for (var childObject in currentObject) {
if (! currentObject.hasOwnProperty(childObject))
continue;
var objectToClone = oData[childObject]['results'][index];
if(objectToClone)
$.extend(true,currentObject[childObject],objectToClone);
}
});
答案 0 :(得分:4)
对不起我之前的评论,我误解了一个问题。这种静谧很有趣,也很有趣。
您正在寻找的表达方式是:
$target_path = ROOT.'/upload/';
$tmp_name = $_FILES['upload']['tmp_name'];
$filename = $_FILES['upload']['name'];
$target_file = $target_path.$filename;
$num = $_POST['num'];
$num_chunks = $_POST['num_chunks'];
move_uploaded_file($tmp_name, $target_file.$num);
// count ammount of uploaded chunks
$chunksUploaded = 0;
for ( $i = 1, i <= $num; $i++ ) {
if ( file_exists( $target_file.$i ) ) {
++$chunksUploaded;
}
}
// and THAT's what you were asking for
// when this triggers - that means your chunks are uploaded
if ($chunksUploaded === $num_chunks) {
/* here you can reassemble chunks together */
for ($i = 1; $i <= $num_chunks; $i++) {
$file = fopen($target_file.$i, 'rb');
$buff = fread($file, 2097152);
fclose($file);
$final = fopen($target_file, 'ab');
$write = fwrite($final, $buff);
fclose($final);
unlink($target_file.$i);
}
}
必须提到这一点:
我的版本的脆弱点 - 是你期待的文件
&#39; TMP-1&#39;,
&#39; TMP-2&#39;,
&#39; TMP-3&#39;
但是,让我们假设发送&#t; tmp-2&#39;我们被打断了 - tmp-2污染了tmp文件夹,它将干扰未来使用相同文件名的上传 - 这将是一个沉睡的炸弹。
要反击 - 你必须找到一种方法将tmp改为更原始的东西。
&#39; TMP-ABCew-1&#39;,
&#39; TMP-ABCew-2&#39;,
&#39; TMP-ABCew-3&#39;
有点好 - 在哪里&#39; ABCew&#39;可以被称为&#39; chunksSessionId&#39; - 您在发送POST时提供它,您随机进行。然而,碰撞是可能的 - 随机名称的空间耗尽。您可以为方程式添加时间 - 例如 - 您可以看到
&#39; TMP-ABCew-2016-03-17-00-11-22--1&#39;,
&#39; TMP-ABCew-2016-03-17-00-11-22--2&#39;,
&#39; TMP-ABCew-2016-03-17-00-11-22--3&#39;
更具抗冲突性,但很难实现 - 这里有一整套蠕虫 - 客户端日期和时间由客户端控制,可能会被欺骗 - 这些数据不可靠。
因此,使tmp-name独特是一项复杂的任务。设计一个让它变得可靠的系统 - 这是一个有趣的问题^ ^你可以使用它。