Merge branch 'master' into fixing-4011-master
Conflicts: lib/connector/sabre/directory.php
This commit is contained in:
commit
c62dc4fa80
|
@ -48,6 +48,7 @@ $defaults = new OC_Defaults();
|
|||
$server->addPlugin(new Sabre_DAV_Auth_Plugin($authBackend, $defaults->getName()));
|
||||
$server->addPlugin(new Sabre_DAV_Locks_Plugin($lockBackend));
|
||||
$server->addPlugin(new Sabre_DAV_Browser_Plugin(false)); // Show something in the Browser, but no upload
|
||||
$server->addPlugin(new OC_Connector_Sabre_AbortedUploadDetectionPlugin());
|
||||
$server->addPlugin(new OC_Connector_Sabre_QuotaPlugin());
|
||||
$server->addPlugin(new OC_Connector_Sabre_MaintenancePlugin());
|
||||
|
||||
|
|
|
@ -0,0 +1,101 @@
|
|||
<?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.
|
||||
*/
|
||||
class OC_Connector_Sabre_AbortedUploadDetectionPlugin extends Sabre_DAV_ServerPlugin {
|
||||
|
||||
/**
|
||||
* Reference to main server object
|
||||
*
|
||||
* @var Sabre_DAV_Server
|
||||
*/
|
||||
private $server;
|
||||
|
||||
/**
|
||||
* is kept public to allow overwrite for unit testing
|
||||
*
|
||||
* @var \OC\Files\View
|
||||
*/
|
||||
public $fileView;
|
||||
|
||||
/**
|
||||
* This initializes the plugin.
|
||||
*
|
||||
* This function is called by Sabre_DAV_Server, after
|
||||
* addPlugin is called.
|
||||
*
|
||||
* This method should set up the requires event subscriptions.
|
||||
*
|
||||
* @param Sabre_DAV_Server $server
|
||||
*/
|
||||
public function initialize(Sabre_DAV_Server $server) {
|
||||
|
||||
$this->server = $server;
|
||||
|
||||
$server->subscribeEvent('afterCreateFile', array($this, 'verifyContentLength'), 10);
|
||||
$server->subscribeEvent('afterWriteContent', array($this, 'verifyContentLength'), 10);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $filePath
|
||||
* @param Sabre_DAV_INode $node
|
||||
* @throws Sabre_DAV_Exception_BadRequest
|
||||
*/
|
||||
public function verifyContentLength($filePath, Sabre_DAV_INode $node = null) {
|
||||
|
||||
// ownCloud chunked upload will be handled in its own plugin
|
||||
$chunkHeader = $this->server->httpRequest->getHeader('OC-Chunked');
|
||||
if ($chunkHeader) {
|
||||
return;
|
||||
}
|
||||
|
||||
// compare expected and actual size
|
||||
$expected = $this->getLength();
|
||||
if (!$expected) {
|
||||
return;
|
||||
}
|
||||
$actual = $this->getFileView()->filesize($filePath);
|
||||
if ($actual != $expected) {
|
||||
$this->getFileView()->unlink($filePath);
|
||||
throw new Sabre_DAV_Exception_BadRequest('expected filesize ' . $expected . ' got ' . $actual);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string
|
||||
*/
|
||||
public function getLength()
|
||||
{
|
||||
$req = $this->server->httpRequest;
|
||||
$length = $req->getHeader('X-Expected-Entity-Length');
|
||||
if (!$length) {
|
||||
$length = $req->getHeader('Content-Length');
|
||||
}
|
||||
|
||||
return $length;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return \OC\Files\View
|
||||
*/
|
||||
public function getFileView()
|
||||
{
|
||||
if (is_null($this->fileView)) {
|
||||
// initialize fileView
|
||||
$this->fileView = \OC\Files\Filesystem::getView();
|
||||
}
|
||||
|
||||
return $this->fileView;
|
||||
}
|
||||
}
|
|
@ -57,6 +57,7 @@ class OC_Connector_Sabre_Directory extends OC_Connector_Sabre_Node implements Sa
|
|||
$path = $this->path . '/' . $name;
|
||||
$node = new OC_Connector_Sabre_File($path);
|
||||
return $node->put($data);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -90,19 +90,6 @@ class OC_Connector_Sabre_File extends OC_Connector_Sabre_Node implements Sabre_D
|
|||
throw new Sabre_DAV_Exception_Forbidden();
|
||||
}
|
||||
|
||||
//detect aborted upload
|
||||
if (isset ($_SERVER['REQUEST_METHOD']) && $_SERVER['REQUEST_METHOD'] === 'PUT' ) {
|
||||
if (isset($_SERVER['CONTENT_LENGTH'])) {
|
||||
$expected = $_SERVER['CONTENT_LENGTH'];
|
||||
$actual = $fs->filesize($partpath);
|
||||
if ($actual != $expected) {
|
||||
$fs->unlink($partpath);
|
||||
throw new Sabre_DAV_Exception_BadRequest(
|
||||
'expected filesize ' . $expected . ' got ' . $actual);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// rename to correct path
|
||||
$renameOkay = $fs->rename($partpath, $this->path);
|
||||
$fileExists = $fs->file_exists($this->path);
|
||||
|
@ -173,7 +160,7 @@ class OC_Connector_Sabre_File extends OC_Connector_Sabre_Node implements Sabre_D
|
|||
*
|
||||
* An ETag is a unique identifier representing the current version of the
|
||||
* file. If the file changes, the ETag MUST change. The ETag is an
|
||||
* arbritrary string, but MUST be surrounded by double-quotes.
|
||||
* arbitrary string, but MUST be surrounded by double-quotes.
|
||||
*
|
||||
* Return null if the ETag can not effectively be determined
|
||||
*
|
||||
|
|
|
@ -0,0 +1,97 @@
|
|||
<?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 Test_OC_Connector_Sabre_AbortedUploadDetectionPlugin extends PHPUnit_Framework_TestCase {
|
||||
|
||||
/**
|
||||
* @var Sabre_DAV_Server
|
||||
*/
|
||||
private $server;
|
||||
|
||||
/**
|
||||
* @var OC_Connector_Sabre_AbortedUploadDetectionPlugin
|
||||
*/
|
||||
private $plugin;
|
||||
|
||||
public function setUp() {
|
||||
$this->server = new Sabre_DAV_Server();
|
||||
$this->plugin = new OC_Connector_Sabre_AbortedUploadDetectionPlugin();
|
||||
$this->plugin->initialize($this->server);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider lengthProvider
|
||||
*/
|
||||
public function testLength($expected, $headers)
|
||||
{
|
||||
$this->server->httpRequest = new Sabre_HTTP_Request($headers);
|
||||
$length = $this->plugin->getLength();
|
||||
$this->assertEquals($expected, $length);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider verifyContentLengthProvider
|
||||
*/
|
||||
public function testVerifyContentLength($fileSize, $headers)
|
||||
{
|
||||
$this->plugin->fileView = $this->buildFileViewMock($fileSize);
|
||||
|
||||
$this->server->httpRequest = new Sabre_HTTP_Request($headers);
|
||||
$this->plugin->verifyContentLength('foo.txt');
|
||||
$this->assertTrue(true);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider verifyContentLengthFailedProvider
|
||||
* @expectedException Sabre_DAV_Exception_BadRequest
|
||||
*/
|
||||
public function testVerifyContentLengthFailed($fileSize, $headers)
|
||||
{
|
||||
$this->plugin->fileView = $this->buildFileViewMock($fileSize);
|
||||
|
||||
// we expect unlink to be called
|
||||
$this->plugin->fileView->expects($this->once())->method('unlink');
|
||||
|
||||
|
||||
$this->server->httpRequest = new Sabre_HTTP_Request($headers);
|
||||
$this->plugin->verifyContentLength('foo.txt');
|
||||
}
|
||||
|
||||
public function verifyContentLengthProvider() {
|
||||
return array(
|
||||
array(1024, array()),
|
||||
array(1024, array('HTTP_X_EXPECTED_ENTITY_LENGTH' => '1024')),
|
||||
array(512, array('HTTP_CONTENT_LENGTH' => '512')),
|
||||
);
|
||||
}
|
||||
|
||||
public function verifyContentLengthFailedProvider() {
|
||||
return array(
|
||||
array(1025, array('HTTP_X_EXPECTED_ENTITY_LENGTH' => '1024')),
|
||||
array(525, array('HTTP_CONTENT_LENGTH' => '512')),
|
||||
);
|
||||
}
|
||||
|
||||
public function lengthProvider() {
|
||||
return array(
|
||||
array(null, array()),
|
||||
array(1024, array('HTTP_X_EXPECTED_ENTITY_LENGTH' => '1024')),
|
||||
array(512, array('HTTP_CONTENT_LENGTH' => '512')),
|
||||
array(2048, array('HTTP_X_EXPECTED_ENTITY_LENGTH' => '2048', 'HTTP_CONTENT_LENGTH' => '1024')),
|
||||
);
|
||||
}
|
||||
|
||||
private function buildFileViewMock($fileSize) {
|
||||
// mock filesysten
|
||||
$view = $this->getMock('\OC\Files\View', array('filesize', 'unlink'), array(), '', FALSE);
|
||||
$view->expects($this->any())->method('filesize')->withAnyParameters()->will($this->returnValue($fileSize));
|
||||
|
||||
return $view;
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue