summaryrefslogtreecommitdiffstats
path: root/apps/dav/lib
diff options
context:
space:
mode:
authorMorris Jobke <hey@morrisjobke.de>2018-01-03 16:18:24 +0100
committerGitHub <noreply@github.com>2018-01-03 16:18:24 +0100
commit876238ce8ba2a9b3c65e53f88806db8e665d6248 (patch)
tree22fe1537b93fabb7ea1f2d556a56047a2e6c6eb5 /apps/dav/lib
parentee0653d68b117ac586841465840e629a7c1a487f (diff)
parentec8bf5335618b0b60737ff9a8945d2a835933259 (diff)
downloadnextcloud-server-876238ce8ba2a9b3c65e53f88806db8e665d6248.tar.gz
nextcloud-server-876238ce8ba2a9b3c65e53f88806db8e665d6248.zip
Merge pull request #7533 from nextcloud/oc-28545-handle-oc-total-length-in-new-chunking
[oc] Handle OC-Total-Length in new chunking
Diffstat (limited to 'apps/dav/lib')
-rw-r--r--apps/dav/lib/Connector/Sabre/File.php12
-rw-r--r--apps/dav/lib/Connector/Sabre/FilesPlugin.php54
-rw-r--r--apps/dav/lib/Connector/Sabre/Node.php14
-rw-r--r--apps/dav/lib/Server.php2
-rw-r--r--apps/dav/lib/Upload/ChunkingPlugin.php106
5 files changed, 126 insertions, 62 deletions
diff --git a/apps/dav/lib/Connector/Sabre/File.php b/apps/dav/lib/Connector/Sabre/File.php
index 32cc8b7adeb..597e6ebef90 100644
--- a/apps/dav/lib/Connector/Sabre/File.php
+++ b/apps/dav/lib/Connector/Sabre/File.php
@@ -589,18 +589,6 @@ class File extends Node implements IFile {
throw new \Sabre\DAV\Exception($e->getMessage(), 0, $e);
}
- private function sanitizeMtime($mtimeFromRequest) {
- // In PHP 5.X "is_numeric" returns true for strings in hexadecimal
- // notation. This is no longer the case in PHP 7.X, so this check
- // ensures that strings with hexadecimal notations fail too in PHP 5.X.
- $isHexadecimal = is_string($mtimeFromRequest) && preg_match('/^\s*0[xX]/', $mtimeFromRequest);
- if ($isHexadecimal || !is_numeric($mtimeFromRequest)) {
- throw new \InvalidArgumentException('X-OC-MTime header must be an integer (unix timestamp).');
- }
-
- return intval($mtimeFromRequest);
- }
-
/**
* Get the checksum for this file
*
diff --git a/apps/dav/lib/Connector/Sabre/FilesPlugin.php b/apps/dav/lib/Connector/Sabre/FilesPlugin.php
index daf9d2d2112..a3a0893259b 100644
--- a/apps/dav/lib/Connector/Sabre/FilesPlugin.php
+++ b/apps/dav/lib/Connector/Sabre/FilesPlugin.php
@@ -32,7 +32,7 @@
namespace OCA\DAV\Connector\Sabre;
-use OC\Files\View;
+use OC\AppFramework\Http\Request;
use OCP\Files\ForbiddenException;
use OCP\IPreview;
use Sabre\DAV\Exception\Forbidden;
@@ -47,7 +47,6 @@ use \Sabre\HTTP\ResponseInterface;
use OCP\Files\StorageNotAvailableException;
use OCP\IConfig;
use OCP\IRequest;
-use OCA\DAV\Upload\FutureFile;
class FilesPlugin extends ServerPlugin {
@@ -91,11 +90,6 @@ class FilesPlugin extends ServerPlugin {
private $isPublic;
/**
- * @var View
- */
- private $fileView;
-
- /**
* @var bool
*/
private $downloadAttachment;
@@ -183,7 +177,6 @@ class FilesPlugin extends ServerPlugin {
}
});
$this->server->on('beforeMove', [$this, 'checkMove']);
- $this->server->on('beforeMove', [$this, 'beforeMoveFutureFile']);
}
/**
@@ -258,9 +251,9 @@ class FilesPlugin extends ServerPlugin {
$filename = $node->getName();
if ($this->request->isUserAgent(
[
- \OC\AppFramework\Http\Request::USER_AGENT_IE,
- \OC\AppFramework\Http\Request::USER_AGENT_ANDROID_MOBILE_CHROME,
- \OC\AppFramework\Http\Request::USER_AGENT_FREEBOX,
+ Request::USER_AGENT_IE,
+ Request::USER_AGENT_ANDROID_MOBILE_CHROME,
+ Request::USER_AGENT_FREEBOX,
])) {
$response->addHeader('Content-Disposition', 'attachment; filename="' . rawurlencode($filename) . '"');
} else {
@@ -461,43 +454,4 @@ class FilesPlugin extends ServerPlugin {
}
}
}
-
- /**
- * Move handler for future file.
- *
- * This overrides the default move behavior to prevent Sabre
- * to delete the target file before moving. Because deleting would
- * lose the file id and metadata.
- *
- * @param string $path source path
- * @param string $destination destination path
- * @return bool|void false to stop handling, void to skip this handler
- */
- public function beforeMoveFutureFile($path, $destination) {
- $sourceNode = $this->tree->getNodeForPath($path);
- if (!$sourceNode instanceof FutureFile) {
- // skip handling as the source is not a chunked FutureFile
- return;
- }
-
- if (!$this->tree->nodeExists($destination)) {
- // skip and let the default handler do its work
- return;
- }
-
- // do a move manually, skipping Sabre's default "delete" for existing nodes
- $this->tree->move($path, $destination);
-
- // trigger all default events (copied from CorePlugin::move)
- $this->server->emit('afterMove', [$path, $destination]);
- $this->server->emit('afterUnbind', [$path]);
- $this->server->emit('afterBind', [$destination]);
-
- $response = $this->server->httpResponse;
- $response->setHeader('Content-Length', '0');
- $response->setStatus(204);
-
- return false;
- }
-
}
diff --git a/apps/dav/lib/Connector/Sabre/Node.php b/apps/dav/lib/Connector/Sabre/Node.php
index 979336d86fe..b8a8209129c 100644
--- a/apps/dav/lib/Connector/Sabre/Node.php
+++ b/apps/dav/lib/Connector/Sabre/Node.php
@@ -165,6 +165,7 @@ abstract class Node implements \Sabre\DAV\INode {
* Even if the modification time is set to a custom value the access time is set to now.
*/
public function touch($mtime) {
+ $mtime = $this->sanitizeMtime($mtime);
$this->fileView->touch($this->path, $mtime);
$this->refreshInfo();
}
@@ -358,4 +359,17 @@ abstract class Node implements \Sabre\DAV\INode {
public function getFileInfo() {
return $this->info;
}
+
+ protected function sanitizeMtime($mtimeFromRequest) {
+ // In PHP 5.X "is_numeric" returns true for strings in hexadecimal
+ // notation. This is no longer the case in PHP 7.X, so this check
+ // ensures that strings with hexadecimal notations fail too in PHP 5.X.
+ $isHexadecimal = is_string($mtimeFromRequest) && preg_match('/^\s*0[xX]/', $mtimeFromRequest);
+ if ($isHexadecimal || !is_numeric($mtimeFromRequest)) {
+ throw new \InvalidArgumentException('X-OC-MTime header must be an integer (unix timestamp).');
+ }
+
+ return intval($mtimeFromRequest);
+ }
+
}
diff --git a/apps/dav/lib/Server.php b/apps/dav/lib/Server.php
index 4e8a9fd0a51..dfa19467e5e 100644
--- a/apps/dav/lib/Server.php
+++ b/apps/dav/lib/Server.php
@@ -55,6 +55,7 @@ use OCA\DAV\DAV\CustomPropertiesBackend;
use OCA\DAV\Connector\Sabre\QuotaPlugin;
use OCA\DAV\Files\BrowserErrorPagePlugin;
use OCA\DAV\SystemTag\SystemTagPlugin;
+use OCA\DAV\Upload\ChunkingPlugin;
use OCP\IRequest;
use OCP\SabrePluginEvent;
use Sabre\CardDAV\VCFExportPlugin;
@@ -171,6 +172,7 @@ class Server {
));
$this->server->addPlugin(new CopyEtagHeaderPlugin());
+ $this->server->addPlugin(new ChunkingPlugin());
// allow setup of additional plugins
$dispatcher->dispatch('OCA\DAV\Connector\Sabre::addPlugin', $event);
diff --git a/apps/dav/lib/Upload/ChunkingPlugin.php b/apps/dav/lib/Upload/ChunkingPlugin.php
new file mode 100644
index 00000000000..5768f53c2b4
--- /dev/null
+++ b/apps/dav/lib/Upload/ChunkingPlugin.php
@@ -0,0 +1,106 @@
+<?php
+/**
+ * @author Thomas Müller <thomas.mueller@tmit.eu>
+ *
+ * @copyright Copyright (c) 2017, ownCloud GmbH
+ * @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/>
+ *
+ */
+
+
+namespace OCA\DAV\Upload;
+
+
+use OCA\DAV\Connector\Sabre\File;
+use Sabre\DAV\Exception\BadRequest;
+use Sabre\DAV\Server;
+use Sabre\DAV\ServerPlugin;
+
+class ChunkingPlugin extends ServerPlugin {
+
+ /** @var Server */
+ private $server;
+ /** @var FutureFile */
+ private $sourceNode;
+
+ /**
+ * @inheritdoc
+ */
+ function initialize(Server $server) {
+ $server->on('beforeMove', [$this, 'beforeMove']);
+ $this->server = $server;
+ }
+
+ /**
+ * @param string $sourcePath source path
+ * @param string $destination destination path
+ */
+ function beforeMove($sourcePath, $destination) {
+ $this->sourceNode = $this->server->tree->getNodeForPath($sourcePath);
+ if (!$this->sourceNode instanceof FutureFile) {
+ // skip handling as the source is not a chunked FutureFile
+ return;
+ }
+
+ $this->verifySize();
+ return $this->performMove($sourcePath, $destination);
+ }
+
+ /**
+ * Move handler for future file.
+ *
+ * This overrides the default move behavior to prevent Sabre
+ * to delete the target file before moving. Because deleting would
+ * lose the file id and metadata.
+ *
+ * @param string $path source path
+ * @param string $destination destination path
+ * @return bool|void false to stop handling, void to skip this handler
+ */
+ public function performMove($path, $destination) {
+ if (!$this->server->tree->nodeExists($destination)) {
+ // skip and let the default handler do its work
+ return;
+ }
+
+ // do a move manually, skipping Sabre's default "delete" for existing nodes
+ $this->server->tree->move($path, $destination);
+
+ // trigger all default events (copied from CorePlugin::move)
+ $this->server->emit('afterMove', [$path, $destination]);
+ $this->server->emit('afterUnbind', [$path]);
+ $this->server->emit('afterBind', [$destination]);
+
+ $response = $this->server->httpResponse;
+ $response->setHeader('Content-Length', '0');
+ $response->setStatus(204);
+
+ return false;
+ }
+
+ /**
+ * @throws BadRequest
+ */
+ private function verifySize() {
+ $expectedSize = $this->server->httpRequest->getHeader('OC-Total-Length');
+ if ($expectedSize === null) {
+ return;
+ }
+ $actualSize = $this->sourceNode->getSize();
+ if ((int)$expectedSize !== $actualSize) {
+ throw new BadRequest("Chunks on server do not sum up to $expectedSize but to $actualSize bytes");
+ }
+ }
+}