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.
|
|
|
|
*/
|
|
|
|
class OC_Connector_Sabre_AbortedUploadDetectionPlugin extends Sabre_DAV_ServerPlugin {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reference to main server object
|
|
|
|
*
|
|
|
|
* @var Sabre_DAV_Server
|
|
|
|
*/
|
|
|
|
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.
|
|
|
|
*
|
|
|
|
* 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 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
|
2013-09-25 19:28:45 +04:00
|
|
|
* @param Sabre_DAV_INode $node
|
|
|
|
* @throws Sabre_DAV_Exception_BadRequest
|
|
|
|
*/
|
|
|
|
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);
|
2013-09-25 19:00:20 +04:00
|
|
|
throw new Sabre_DAV_Exception_BadRequest('expected filesize ' . $expected . ' got ' . $actual);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @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;
|
|
|
|
}
|
|
|
|
}
|