Fix quota calculation on new dav upload endpoint
Signed-off-by: Roeland Jago Douma <roeland@famdouma.nl>
This commit is contained in:
parent
d659199ff5
commit
ab63c89ab3
|
@ -25,10 +25,14 @@
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
namespace OCA\DAV\Connector\Sabre;
|
namespace OCA\DAV\Connector\Sabre;
|
||||||
|
use OCA\DAV\Files\FilesHome;
|
||||||
|
use OCA\DAV\Upload\FutureFile;
|
||||||
|
use OCA\DAV\Upload\UploadFolder;
|
||||||
use OCP\Files\FileInfo;
|
use OCP\Files\FileInfo;
|
||||||
use OCP\Files\StorageNotAvailableException;
|
use OCP\Files\StorageNotAvailableException;
|
||||||
use Sabre\DAV\Exception\InsufficientStorage;
|
use Sabre\DAV\Exception\InsufficientStorage;
|
||||||
use Sabre\DAV\Exception\ServiceUnavailable;
|
use Sabre\DAV\Exception\ServiceUnavailable;
|
||||||
|
use Sabre\DAV\INode;
|
||||||
use Sabre\HTTP\URLUtil;
|
use Sabre\HTTP\URLUtil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -40,9 +44,7 @@ use Sabre\HTTP\URLUtil;
|
||||||
*/
|
*/
|
||||||
class QuotaPlugin extends \Sabre\DAV\ServerPlugin {
|
class QuotaPlugin extends \Sabre\DAV\ServerPlugin {
|
||||||
|
|
||||||
/**
|
/** @var \OC\Files\View */
|
||||||
* @var \OC\Files\View
|
|
||||||
*/
|
|
||||||
private $view;
|
private $view;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -74,26 +76,86 @@ class QuotaPlugin extends \Sabre\DAV\ServerPlugin {
|
||||||
|
|
||||||
$this->server = $server;
|
$this->server = $server;
|
||||||
|
|
||||||
$server->on('beforeWriteContent', array($this, 'checkQuota'), 10);
|
$server->on('beforeWriteContent', [$this, 'beforeWriteContent'], 10);
|
||||||
$server->on('beforeCreateFile', array($this, 'checkQuota'), 10);
|
$server->on('beforeCreateFile', [$this, 'beforeCreateFile'], 10);
|
||||||
|
$server->on('beforeMove', [$this, 'beforeMove'], 10);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check quota before creating file
|
||||||
|
*
|
||||||
|
* @param string $uri target file URI
|
||||||
|
* @param resource $data data
|
||||||
|
* @param INode $parent Sabre Node
|
||||||
|
* @param bool $modified modified
|
||||||
|
*/
|
||||||
|
public function beforeCreateFile($uri, $data, INode $parent, $modified) {
|
||||||
|
if (!$parent instanceof Node) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->checkQuota($parent->getPath() . '/' . basename($uri));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check quota before writing content
|
||||||
|
*
|
||||||
|
* @param string $uri target file URI
|
||||||
|
* @param INode $node Sabre Node
|
||||||
|
* @param resource $data data
|
||||||
|
* @param bool $modified modified
|
||||||
|
*/
|
||||||
|
public function beforeWriteContent($uri, INode $node, $data, $modified) {
|
||||||
|
if (!$node instanceof Node) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->checkQuota($node->getPath());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if we're moving a Futurefile in which case we need to check
|
||||||
|
* the quota on the target destination.
|
||||||
|
*
|
||||||
|
* @param string $source source path
|
||||||
|
* @param string $destination destination path
|
||||||
|
*/
|
||||||
|
public function beforeMove($source, $destination) {
|
||||||
|
$sourceNode = $this->server->tree->getNodeForPath($source);
|
||||||
|
if (!$sourceNode instanceof FutureFile) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// get target node for proper path conversion
|
||||||
|
if ($this->server->tree->nodeExists($destination)) {
|
||||||
|
$destinationNode = $this->server->tree->getNodeForPath($destination);
|
||||||
|
$path = $destinationNode->getPath();
|
||||||
|
} else {
|
||||||
|
$parentNode = $this->server->tree->getNodeForPath(dirname($destination));
|
||||||
|
$path = $parentNode->getPath();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->checkQuota($path, $sourceNode->getSize());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method is called before any HTTP method and validates there is enough free space to store the file
|
* This method is called before any HTTP method and validates there is enough free space to store the file
|
||||||
*
|
*
|
||||||
* @param string $uri
|
* @param string $path relative to the users home
|
||||||
|
* @param int $length
|
||||||
* @throws InsufficientStorage
|
* @throws InsufficientStorage
|
||||||
* @return bool
|
* @return bool
|
||||||
*/
|
*/
|
||||||
public function checkQuota($uri) {
|
public function checkQuota($path, $length = null) {
|
||||||
|
if ($length === null) {
|
||||||
$length = $this->getLength();
|
$length = $this->getLength();
|
||||||
if ($length) {
|
|
||||||
if (substr($uri, 0, 1) !== '/') {
|
|
||||||
$uri = '/' . $uri;
|
|
||||||
}
|
}
|
||||||
list($parentUri, $newName) = \Sabre\Uri\split($uri);
|
|
||||||
if(is_null($parentUri)) {
|
if ($length) {
|
||||||
$parentUri = '';
|
list($parentPath, $newName) = \Sabre\Uri\split($path);
|
||||||
|
if(is_null($parentPath)) {
|
||||||
|
$parentPath = '';
|
||||||
}
|
}
|
||||||
$req = $this->server->httpRequest;
|
$req = $this->server->httpRequest;
|
||||||
if ($req->getHeader('OC-Chunked')) {
|
if ($req->getHeader('OC-Chunked')) {
|
||||||
|
@ -103,9 +165,9 @@ class QuotaPlugin extends \Sabre\DAV\ServerPlugin {
|
||||||
// there is still enough space for the remaining chunks
|
// there is still enough space for the remaining chunks
|
||||||
$length -= $chunkHandler->getCurrentSize();
|
$length -= $chunkHandler->getCurrentSize();
|
||||||
// use target file name for free space check in case of shared files
|
// use target file name for free space check in case of shared files
|
||||||
$uri = rtrim($parentUri, '/') . '/' . $info['name'];
|
$path = rtrim($parentPath, '/') . '/' . $info['name'];
|
||||||
}
|
}
|
||||||
$freeSpace = $this->getFreeSpace($uri);
|
$freeSpace = $this->getFreeSpace($path);
|
||||||
if ($freeSpace !== FileInfo::SPACE_UNKNOWN && $freeSpace !== FileInfo::SPACE_UNLIMITED && $length > $freeSpace) {
|
if ($freeSpace !== FileInfo::SPACE_UNKNOWN && $freeSpace !== FileInfo::SPACE_UNLIMITED && $length > $freeSpace) {
|
||||||
if (isset($chunkHandler)) {
|
if (isset($chunkHandler)) {
|
||||||
$chunkHandler->cleanup();
|
$chunkHandler->cleanup();
|
||||||
|
|
|
@ -161,7 +161,7 @@ class ServerFactory {
|
||||||
!$this->config->getSystemValue('debug', false)
|
!$this->config->getSystemValue('debug', false)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
$server->addPlugin(new \OCA\DAV\Connector\Sabre\QuotaPlugin($view));
|
$server->addPlugin(new \OCA\DAV\Connector\Sabre\QuotaPlugin($view, true));
|
||||||
|
|
||||||
if($this->userSession->isLoggedIn()) {
|
if($this->userSession->isLoggedIn()) {
|
||||||
$server->addPlugin(new \OCA\DAV\Connector\Sabre\TagsPlugin($objectTree, $this->tagManager));
|
$server->addPlugin(new \OCA\DAV\Connector\Sabre\TagsPlugin($objectTree, $this->tagManager));
|
||||||
|
|
|
@ -212,7 +212,7 @@ class Server {
|
||||||
);
|
);
|
||||||
if ($view !== null) {
|
if ($view !== null) {
|
||||||
$this->server->addPlugin(
|
$this->server->addPlugin(
|
||||||
new QuotaPlugin($view));
|
new QuotaPlugin($view, false));
|
||||||
}
|
}
|
||||||
$this->server->addPlugin(
|
$this->server->addPlugin(
|
||||||
new TagsPlugin(
|
new TagsPlugin(
|
||||||
|
|
|
@ -24,7 +24,13 @@
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
namespace OCA\DAV\Tests\unit\Connector\Sabre;
|
namespace OCA\DAV\Tests\unit\Connector\Sabre;
|
||||||
|
use OC\Files\View;
|
||||||
|
use OCA\DAV\Connector\Sabre\Directory;
|
||||||
|
use OCA\DAV\Connector\Sabre\QuotaPlugin;
|
||||||
|
use OCA\DAV\Files\FilesHome;
|
||||||
use OCP\Files\FileInfo;
|
use OCP\Files\FileInfo;
|
||||||
|
use Sabre\DAV\Exception\InsufficientStorage;
|
||||||
|
use Sabre\DAV\Tree;
|
||||||
use Test\TestCase;
|
use Test\TestCase;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -44,7 +50,7 @@ class QuotaPluginTest extends TestCase {
|
||||||
private function init($quota, $checkedPath = '') {
|
private function init($quota, $checkedPath = '') {
|
||||||
$view = $this->buildFileViewMock($quota, $checkedPath);
|
$view = $this->buildFileViewMock($quota, $checkedPath);
|
||||||
$this->server = new \Sabre\DAV\Server();
|
$this->server = new \Sabre\DAV\Server();
|
||||||
$this->plugin = $this->getMockBuilder('\OCA\DAV\Connector\Sabre\QuotaPlugin')
|
$this->plugin = $this->getMockBuilder(QuotaPlugin::class)
|
||||||
->setConstructorArgs([$view])
|
->setConstructorArgs([$view])
|
||||||
->setMethods(['getFileChunking'])
|
->setMethods(['getFileChunking'])
|
||||||
->getMock();
|
->getMock();
|
||||||
|
@ -224,7 +230,7 @@ class QuotaPluginTest extends TestCase {
|
||||||
|
|
||||||
private function buildFileViewMock($quota, $checkedPath) {
|
private function buildFileViewMock($quota, $checkedPath) {
|
||||||
// mock filesysten
|
// mock filesysten
|
||||||
$view = $this->getMockBuilder('\OC\Files\View')
|
$view = $this->getMockBuilder(View::class)
|
||||||
->setMethods(['free_space'])
|
->setMethods(['free_space'])
|
||||||
->disableOriginalConstructor()
|
->disableOriginalConstructor()
|
||||||
->getMock();
|
->getMock();
|
||||||
|
@ -235,5 +241,4 @@ class QuotaPluginTest extends TestCase {
|
||||||
|
|
||||||
return $view;
|
return $view;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue