2013-09-25 19:00:20 +04:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Copyright (c) 2013 Thomas Müller <thomas.mueller@tmit.eu>
|
|
|
|
* This file is licensed under the Affero General Public License version 3 or
|
|
|
|
* later.
|
|
|
|
* See the COPYING-README file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class OC_Connector_Sabre_AbortedUploadDetectionPlugin
|
|
|
|
*
|
|
|
|
* This plugin will verify if the uploaded data has been stored completely.
|
|
|
|
* This is done by comparing the content length of the request with the file size on storage.
|
|
|
|
*/
|
2014-01-09 17:25:48 +04:00
|
|
|
class OC_Connector_Sabre_AbortedUploadDetectionPlugin extends \Sabre\DAV\ServerPlugin {
|
2013-09-25 19:00:20 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Reference to main server object
|
|
|
|
*
|
2014-01-09 17:25:48 +04:00
|
|
|
* @var \Sabre\DAV\Server
|
2013-09-25 19:00:20 +04:00
|
|
|
*/
|
|
|
|
private $server;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \OC\Files\View
|
|
|
|
*/
|
2014-02-25 19:23:09 +04:00
|
|
|
private $fileView;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param \OC\Files\View $view
|
|
|
|
*/
|
|
|
|
public function __construct($view) {
|
|
|
|
$this->fileView = $view;
|
|
|
|
}
|
2013-09-25 19:00:20 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This initializes the plugin.
|
|
|
|
*
|
2014-01-09 17:25:48 +04:00
|
|
|
* This function is called by \Sabre\DAV\Server, after
|
2013-09-25 19:00:20 +04:00
|
|
|
* addPlugin is called.
|
|
|
|
*
|
|
|
|
* This method should set up the requires event subscriptions.
|
|
|
|
*
|
2014-01-09 17:25:48 +04:00
|
|
|
* @param \Sabre\DAV\Server $server
|
2013-09-25 19:00:20 +04:00
|
|
|
*/
|
2014-01-09 17:25:48 +04:00
|
|
|
public function initialize(\Sabre\DAV\Server $server) {
|
2013-09-25 19:00:20 +04:00
|
|
|
|
|
|
|
$this->server = $server;
|
|
|
|
|
2013-09-25 19:28:45 +04:00
|
|
|
$server->subscribeEvent('afterCreateFile', array($this, 'verifyContentLength'), 10);
|
|
|
|
$server->subscribeEvent('afterWriteContent', array($this, 'verifyContentLength'), 10);
|
2013-09-25 19:00:20 +04:00
|
|
|
}
|
|
|
|
|
2013-09-25 19:28:45 +04:00
|
|
|
/**
|
2014-02-19 12:31:54 +04:00
|
|
|
* @param string $filePath
|
2014-01-09 17:25:48 +04:00
|
|
|
* @param \Sabre\DAV\INode $node
|
|
|
|
* @throws \Sabre\DAV\Exception\BadRequest
|
2013-09-25 19:28:45 +04:00
|
|
|
*/
|
2014-01-09 17:25:48 +04:00
|
|
|
public function verifyContentLength($filePath, \Sabre\DAV\INode $node = null) {
|
2013-09-25 19:00:20 +04:00
|
|
|
|
2013-10-04 18:09:07 +04:00
|
|
|
// we should only react on PUT which is used for upload
|
|
|
|
// e.g. with LOCK this will not work, but LOCK uses createFile() as well
|
2014-02-25 19:23:09 +04:00
|
|
|
if ($this->server->httpRequest->getMethod() !== 'PUT') {
|
2013-10-04 18:09:07 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-09-30 12:03:07 +04:00
|
|
|
// ownCloud chunked upload will be handled in its own plugin
|
2013-09-25 19:00:20 +04:00
|
|
|
$chunkHeader = $this->server->httpRequest->getHeader('OC-Chunked');
|
|
|
|
if ($chunkHeader) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// compare expected and actual size
|
|
|
|
$expected = $this->getLength();
|
2013-09-25 19:28:45 +04:00
|
|
|
if (!$expected) {
|
|
|
|
return;
|
|
|
|
}
|
2014-02-25 19:23:09 +04:00
|
|
|
$actual = $this->fileView->filesize($filePath);
|
2013-09-25 19:00:20 +04:00
|
|
|
if ($actual != $expected) {
|
2014-02-25 19:23:09 +04:00
|
|
|
$this->fileView->unlink($filePath);
|
2014-01-09 17:25:48 +04:00
|
|
|
throw new \Sabre\DAV\Exception\BadRequest('expected filesize ' . $expected . ' got ' . $actual);
|
2013-09-25 19:00:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
2014-02-25 19:23:09 +04:00
|
|
|
public function getLength() {
|
2013-09-25 19:00:20 +04:00
|
|
|
$req = $this->server->httpRequest;
|
|
|
|
$length = $req->getHeader('X-Expected-Entity-Length');
|
|
|
|
if (!$length) {
|
|
|
|
$length = $req->getHeader('Content-Length');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $length;
|
|
|
|
}
|
|
|
|
}
|