-
Notifications
You must be signed in to change notification settings - Fork 2.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Initial implementation of the new chunked upload #20118
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,76 @@ | ||
<?php | ||
/** | ||
* @author Thomas Müller <[email protected]> | ||
* | ||
* @copyright Copyright (c) 2016, ownCloud, Inc. | ||
* @license AGPL-3.0 | ||
* | ||
* This code is free software: you can redistribute it and/or modify | ||
* it under the terms of the GNU Affero General Public License, version 3, | ||
* as published by the Free Software Foundation. | ||
* | ||
* This program is distributed in the hope that it will be useful, | ||
* but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
* GNU Affero General Public License for more details. | ||
* | ||
* You should have received a copy of the GNU Affero General Public License, version 3, | ||
* along with this program. If not, see <http://www.gnu.org/licenses/> | ||
* | ||
*/ | ||
|
||
require '../../../../3rdparty/autoload.php'; | ||
|
||
if ($argc !== 6) { | ||
echo "Invalid number of arguments" . PHP_EOL; | ||
exit; | ||
} | ||
|
||
/** | ||
* @param \Sabre\DAV\Client $client | ||
* @param $uploadUrl | ||
* @return mixed | ||
*/ | ||
function request($client, $method, $uploadUrl, $data = null, $headers = []) { | ||
echo "$method $uploadUrl ... "; | ||
$t0 = microtime(true); | ||
$result = $client->request($method, $uploadUrl, $data, $headers); | ||
$t1 = microtime(true); | ||
echo $result['statusCode'] . " - " . ($t1 - $t0) . ' seconds' . PHP_EOL; | ||
if (!in_array($result['statusCode'], [200, 201])) { | ||
echo $result['body'] . PHP_EOL; | ||
} | ||
return $result; | ||
} | ||
|
||
$baseUri = $argv[1]; | ||
$userName = $argv[2]; | ||
$password = $argv[3]; | ||
$file = $argv[4]; | ||
$chunkSize = $argv[5] * 1024 * 1024; | ||
|
||
$client = new \Sabre\DAV\Client([ | ||
'baseUri' => $baseUri, | ||
'userName' => $userName, | ||
'password' => $password | ||
]); | ||
|
||
$transfer = uniqid('transfer', true); | ||
$uploadUrl = "$baseUri/uploads/$userName/$transfer"; | ||
|
||
request($client, 'MKCOL', $uploadUrl); | ||
|
||
$size = filesize($file); | ||
$stream = fopen($file, 'r'); | ||
|
||
$index = 0; | ||
while(!feof($stream)) { | ||
request($client, 'PUT', "$uploadUrl/$index", fread($stream, $chunkSize)); | ||
$index++; | ||
} | ||
|
||
$destination = pathinfo($file, PATHINFO_BASENAME); | ||
//echo "Moving $uploadUrl/.file to it's final destination $baseUri/files/$userName/$destination" . PHP_EOL; | ||
request($client, 'MOVE', "$uploadUrl/.file", null, [ | ||
'Destination' => "$baseUri/files/$userName/$destination" | ||
]); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,234 @@ | ||
<?php | ||
|
||
namespace OCA\DAV\Upload; | ||
|
||
use Sabre\DAV\IFile; | ||
|
||
/** | ||
* Class AssemblyStream | ||
* | ||
* The assembly stream is a virtual stream that wraps multiple chunks. | ||
* Reading from the stream transparently accessed the underlying chunks and | ||
* give a representation as if they were already merged together. | ||
* | ||
* @package OCA\DAV\Upload | ||
*/ | ||
class AssemblyStream implements \Icewind\Streams\File { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can you add a comment here to explain the AssemblyStream to avoid the many possible confusions ? |
||
|
||
/** @var resource */ | ||
private $context; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm reasonably sure this needs to be public There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @icewind1991 can you elaborate? I did not face any issues having this private |
||
|
||
/** @var IFile[] */ | ||
private $nodes; | ||
|
||
/** @var int */ | ||
private $pos = 0; | ||
|
||
/** @var array */ | ||
private $sortedNodes; | ||
|
||
/** @var int */ | ||
private $size; | ||
|
||
/** | ||
* @param string $path | ||
* @param string $mode | ||
* @param int $options | ||
* @param string &$opened_path | ||
* @return bool | ||
*/ | ||
public function stream_open($path, $mode, $options, &$opened_path) { | ||
$this->loadContext('assembly'); | ||
|
||
// sort the nodes | ||
$nodes = $this->nodes; | ||
// http://stackoverflow.com/a/10985500 | ||
@usort($nodes, function(IFile $a, IFile $b) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No idea ;-) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ah - because of that: usort(): Array was modified by the user comparison function |
||
return strcmp($a->getName(), $b->getName()); | ||
}); | ||
$this->nodes = $nodes; | ||
|
||
// build additional information | ||
$this->sortedNodes = []; | ||
$start = 0; | ||
foreach($this->nodes as $node) { | ||
$size = $node->getSize(); | ||
$name = $node->getName(); | ||
$this->sortedNodes[$name] = ['node' => $node, 'start' => $start, 'end' => $start + $size]; | ||
$start += $size; | ||
$this->size = $start; | ||
} | ||
return true; | ||
} | ||
|
||
/** | ||
* @param string $offset | ||
* @param int $whence | ||
* @return bool | ||
*/ | ||
public function stream_seek($offset, $whence = SEEK_SET) { | ||
return false; | ||
} | ||
|
||
/** | ||
* @return int | ||
*/ | ||
public function stream_tell() { | ||
return $this->pos; | ||
} | ||
|
||
/** | ||
* @param int $count | ||
* @return string | ||
*/ | ||
public function stream_read($count) { | ||
|
||
list($node, $posInNode) = $this->getNodeForPosition($this->pos); | ||
if (is_null($node)) { | ||
return null; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. not a string as indicated in PHP Doc |
||
} | ||
$stream = $this->getStream($node); | ||
|
||
fseek($stream, $posInNode); | ||
$data = fread($stream, $count); | ||
$read = strlen($data); | ||
|
||
// update position | ||
$this->pos += $read; | ||
return $data; | ||
} | ||
|
||
/** | ||
* @param string $data | ||
* @return int | ||
*/ | ||
public function stream_write($data) { | ||
return false; | ||
} | ||
|
||
/** | ||
* @param int $option | ||
* @param int $arg1 | ||
* @param int $arg2 | ||
* @return bool | ||
*/ | ||
public function stream_set_option($option, $arg1, $arg2) { | ||
return false; | ||
} | ||
|
||
/** | ||
* @param int $size | ||
* @return bool | ||
*/ | ||
public function stream_truncate($size) { | ||
return false; | ||
} | ||
|
||
/** | ||
* @return array | ||
*/ | ||
public function stream_stat() { | ||
return []; | ||
} | ||
|
||
/** | ||
* @param int $operation | ||
* @return bool | ||
*/ | ||
public function stream_lock($operation) { | ||
return false; | ||
} | ||
|
||
/** | ||
* @return bool | ||
*/ | ||
public function stream_flush() { | ||
return false; | ||
} | ||
|
||
/** | ||
* @return bool | ||
*/ | ||
public function stream_eof() { | ||
return $this->pos >= $this->size; | ||
} | ||
|
||
/** | ||
* @return bool | ||
*/ | ||
public function stream_close() { | ||
return true; | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. PHP Doc wants to return bool not null |
||
|
||
|
||
/** | ||
* Load the source from the stream context and return the context options | ||
* | ||
* @param string $name | ||
* @return array | ||
* @throws \Exception | ||
*/ | ||
protected function loadContext($name) { | ||
$context = stream_context_get_options($this->context); | ||
if (isset($context[$name])) { | ||
$context = $context[$name]; | ||
} else { | ||
throw new \BadMethodCallException('Invalid context, "' . $name . '" options not set'); | ||
} | ||
if (isset($context['nodes']) and is_array($context['nodes'])) { | ||
$this->nodes = $context['nodes']; | ||
} else { | ||
throw new \BadMethodCallException('Invalid context, nodes not set'); | ||
} | ||
return $context; | ||
} | ||
|
||
/** | ||
* @param IFile[] $nodes | ||
* @return resource | ||
* | ||
* @throws \BadMethodCallException | ||
*/ | ||
public static function wrap(array $nodes) { | ||
$context = stream_context_create([ | ||
'assembly' => [ | ||
'nodes' => $nodes] | ||
]); | ||
stream_wrapper_register('assembly', '\OCA\DAV\Upload\AssemblyStream'); | ||
try { | ||
$wrapped = fopen('assembly://', 'r', null, $context); | ||
} catch (\BadMethodCallException $e) { | ||
stream_wrapper_unregister('assembly'); | ||
throw $e; | ||
} | ||
stream_wrapper_unregister('assembly'); | ||
return $wrapped; | ||
} | ||
|
||
/** | ||
* @param $pos | ||
* @return IFile | null | ||
*/ | ||
private function getNodeForPosition($pos) { | ||
foreach($this->sortedNodes as $node) { | ||
if ($pos >= $node['start'] && $pos < $node['end']) { | ||
return [$node['node'], $pos - $node['start']]; | ||
} | ||
} | ||
return null; | ||
} | ||
|
||
/** | ||
* @param IFile $node | ||
* @return resource | ||
*/ | ||
private function getStream(IFile $node) { | ||
$data = $node->get(); | ||
if (is_resource($data)) { | ||
return $data; | ||
} | ||
|
||
return fopen('data://text/plain,' . $data,'r'); | ||
} | ||
|
||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Isn't that a bit to liberal? What if the file we are uploading to does not allow updates?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a valid scenario we need to cover in the second part of the implementation which is know as 'Announcing the upload'
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah OKIDOKI then.