2013-09-25 15:00:20 +00: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.
|
|
|
|
*/
|
|
|
|
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;
|
|
|
|
|
2013-09-25 15:28:45 +00:00
|
|
|
$server->subscribeEvent('afterCreateFile', array($this, 'verifyContentLength'), 10);
|
|
|
|
$server->subscribeEvent('afterWriteContent', array($this, 'verifyContentLength'), 10);
|
2013-09-25 15:00:20 +00:00
|
|
|
}
|
|
|
|
|
2013-09-25 15:28:45 +00:00
|
|
|
/**
|
|
|
|
* @param $filePath
|
|
|
|
* @param Sabre_DAV_INode $node
|
|
|
|
* @throws Sabre_DAV_Exception_BadRequest
|
|
|
|
*/
|
|
|
|
public function verifyContentLength($filePath, Sabre_DAV_INode $node = null) {
|
2013-09-25 15:00:20 +00:00
|
|
|
|
2013-09-30 08:03:07 +00:00
|
|
|
// ownCloud chunked upload will be handled in its own plugin
|
2013-09-25 15:00:20 +00:00
|
|
|
$chunkHeader = $this->server->httpRequest->getHeader('OC-Chunked');
|
|
|
|
if ($chunkHeader) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// compare expected and actual size
|
|
|
|
$expected = $this->getLength();
|
2013-09-25 15:28:45 +00:00
|
|
|
if (!$expected) {
|
|
|
|
return;
|
|
|
|
}
|
2013-09-25 15:00:20 +00:00
|
|
|
$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;
|
|
|
|
}
|
|
|
|
}
|