summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVincent Petry <pvince81@owncloud.com>2015-01-21 16:29:52 +0100
committerVincent Petry <pvince81@owncloud.com>2015-01-23 12:20:54 +0100
commit67f1534e0fd7c14e97fe5b17bd92aa2277520604 (patch)
tree52b863953936295764a52036cebc1162a6507dfd
parent5fb8a4715d6ed34b1d94c5508700f3c488c0f734 (diff)
downloadnextcloud-server-67f1534e0fd7c14e97fe5b17bd92aa2277520604.tar.gz
nextcloud-server-67f1534e0fd7c14e97fe5b17bd92aa2277520604.zip
Call final unlink in trash wrapper's storage
In the case of cross-storage delete, the files are copied to the trash, then deleted. The final delete on the source storage would still reach the trash wrapper. This fix makes forwards that second call to the wrapped storage to make the final delete work. It fixes the issue with remote shares, local shares and external storage. Also, it uses a new function "renameRecursive" that renames the files and preserves the mtimes (like "copy_recursive" did in the past))
-rwxr-xr-xapps/files_encryption/tests/trashbin.php2
-rw-r--r--apps/files_trashbin/lib/storage.php3
-rw-r--r--apps/files_trashbin/lib/trashbin.php43
-rw-r--r--apps/files_trashbin/tests/storage.php132
4 files changed, 178 insertions, 2 deletions
diff --git a/apps/files_encryption/tests/trashbin.php b/apps/files_encryption/tests/trashbin.php
index b759c8e32fd..2704a9752cc 100755
--- a/apps/files_encryption/tests/trashbin.php
+++ b/apps/files_encryption/tests/trashbin.php
@@ -93,6 +93,8 @@ class Trashbin extends TestCase {
// cleanup test user
\OC_User::deleteUser(self::TEST_ENCRYPTION_TRASHBIN_USER1);
+ \OC\Files\Filesystem::getLoader()->removeStorageWrapper('oc_trashbin');
+
parent::tearDownAfterClass();
}
diff --git a/apps/files_trashbin/lib/storage.php b/apps/files_trashbin/lib/storage.php
index aa5d48b5fbe..5036a260d0c 100644
--- a/apps/files_trashbin/lib/storage.php
+++ b/apps/files_trashbin/lib/storage.php
@@ -46,6 +46,9 @@ class Storage extends Wrapper {
if (count($parts) > 3 && $parts[2] === 'files') {
$filesPath = implode('/', array_slice($parts, 3));
$result = \OCA\Files_Trashbin\Trashbin::move2trash($filesPath);
+ // in cross-storage cases the file will be copied
+ // but not deleted, so we delete it here
+ $this->storage->unlink($path);
} else {
$result = $this->storage->unlink($path);
}
diff --git a/apps/files_trashbin/lib/trashbin.php b/apps/files_trashbin/lib/trashbin.php
index f5cebea6b78..1833936ea26 100644
--- a/apps/files_trashbin/lib/trashbin.php
+++ b/apps/files_trashbin/lib/trashbin.php
@@ -166,8 +166,7 @@ class Trashbin {
\OC_FileProxy::$enabled = false;
$trashPath = '/files_trashbin/files/' . $filename . '.d' . $timestamp;
try {
- $sizeOfAddedFiles = $view->filesize('/files/' . $file_path);
- $view->rename('/files/' . $file_path, $trashPath);
+ $sizeOfAddedFiles = self::renameRecursive('/files/'.$file_path, $trashPath, $view);
} catch (\OCA\Files_Trashbin\Exceptions\CopyRecursiveException $e) {
$sizeOfAddedFiles = false;
if ($view->file_exists($trashPath)) {
@@ -807,6 +806,46 @@ class Trashbin {
}
/**
+ * recursive rename a whole directory and preserve timestamps
+ *
+ * @param string $source source path, relative to the users files directory
+ * @param string $destination destination path relative to the users root directoy
+ * @param \OC\Files\View $view file view for the users root directory
+ * @return int
+ * @throws Exceptions\CopyRecursiveException
+ */
+ private static function renameRecursive($source, $destination, \OC\Files\View $view) {
+ $size = 0;
+ if ($view->is_dir($source)) {
+ $view->mkdir($destination);
+ $view->touch($destination, $view->filemtime($source));
+ foreach ($view->getDirectoryContent($source) as $i) {
+ $pathDir = $source . '/' . $i['name'];
+ if ($view->is_dir($pathDir)) {
+ $size += self::renameRecursive($pathDir, $destination . '/' . $i['name'], $view);
+ } else {
+ $size += $view->filesize($pathDir);
+ $mtime = $view->filemtime($pathDir);
+ $result = $view->rename($pathDir, $destination . '/' . $i['name']);
+ if (!$result) {
+ throw new \OCA\Files_Trashbin\Exceptions\CopyRecursiveException();
+ }
+ $view->touch($destination . '/' . $i['name'], $mtime);
+ }
+ }
+ } else {
+ $size += $view->filesize($source);
+ $mtime = $view->filemtime($source);
+ $result = $view->rename($source, $destination);
+ if (!$result) {
+ throw new \OCA\Files_Trashbin\Exceptions\CopyRecursiveException();
+ }
+ $view->touch($destination, $mtime);
+ }
+ return $size;
+ }
+
+ /**
* find all versions which belong to the file we want to restore
*
* @param string $filename name of the file which should be restored
diff --git a/apps/files_trashbin/tests/storage.php b/apps/files_trashbin/tests/storage.php
new file mode 100644
index 00000000000..c340b9d2362
--- /dev/null
+++ b/apps/files_trashbin/tests/storage.php
@@ -0,0 +1,132 @@
+<?php
+/**
+ * Copyright (c) 2015 Vincent Petry <pvince81@owncloud.com>
+ * This file is licensed under the Affero General Public License version 3 or
+ * later.
+ * See the COPYING-README file.
+ */
+
+namespace OCA\Files_trashbin\Tests\Storage;
+
+use OC\Files\Storage\Home;
+use OC\Files\Storage\Temporary;
+use OC\Files\Mount\MountPoint;
+use OC\Files\Filesystem;
+
+class Storage extends \Test\TestCase {
+ /**
+ * @var \OCA\Files_trashbin\Storage
+ */
+ private $wrapper;
+
+ /**
+ * @var \OCP\Files\Storage
+ */
+ private $storage;
+
+ /**
+ * @var string
+ */
+ private $user;
+
+ /**
+ * @var \OC\Files\Storage\Storage
+ **/
+ private $originalStorage;
+
+ /**
+ * @var \OC\Files\View
+ */
+ private $userView;
+
+ protected function setUp() {
+ parent::setUp();
+
+ $this->user = $this->getUniqueId('user');
+ \OC_User::createUser($this->user, $this->user);
+
+ // this will setup the FS
+ $this->loginAsUser($this->user);
+
+ $this->originalStorage = \OC\Files\Filesystem::getStorage('/');
+
+ $mockUser = $this->getMock('\OCP\IUser');
+ $mockUser->expects($this->any())
+ ->method('getHome')
+ ->will($this->returnValue($this->originalStorage->getLocalFolder($this->user)));
+ $mockUser->expects($this->any())
+ ->method('getUID')
+ ->will($this->returnValue($this->user));
+
+ // use temp as root storage so we can wrap it for testing
+ $this->storage = new Home(
+ array('user' => $mockUser)
+ );
+ $this->wrapper = new \OCA\Files_Trashbin\Storage(
+ array(
+ 'storage' => $this->storage,
+ 'mountPoint' => $this->user,
+ )
+ );
+
+ // make room for a new root
+ Filesystem::clearMounts();
+ $rootMount = new MountPoint($this->originalStorage, '');
+ Filesystem::getMountManager()->addMount($rootMount);
+ $homeMount = new MountPoint($this->wrapper, $this->user);
+ Filesystem::getMountManager()->addMount($homeMount);
+
+ $this->userView = new \OC\Files\View('/' . $this->user . '/files/');
+ $this->userView->file_put_contents('test.txt', 'foo');
+ }
+
+ protected function tearDown() {
+ \OC\Files\Filesystem::mount($this->originalStorage, array(), '/');
+ $this->logout();
+ parent::tearDown();
+ }
+
+ public function testSingleStorageDelete() {
+ $this->assertTrue($this->storage->file_exists('files/test.txt'));
+ $this->userView->unlink('test.txt');
+ $this->storage->getScanner()->scan('');
+ $this->assertFalse($this->userView->getFileInfo('test.txt'));
+ $this->assertFalse($this->storage->file_exists('files/test.txt'));
+
+ // check if file is in trashbin
+ $rootView = new \OC\Files\View('/');
+ $results = $rootView->getDirectoryContent($this->user . '/files_trashbin/files/');
+ $this->assertEquals(1, count($results));
+ $name = $results[0]->getName();
+ $this->assertEquals('test.txt', substr($name, 0, strrpos($name, '.')));
+ }
+
+ public function testCrossStorageDelete() {
+ $storage2 = new Temporary(array());
+ $wrapper2 = new \OCA\Files_Trashbin\Storage(
+ array(
+ 'storage' => $storage2,
+ 'mountPoint' => $this->user . '/files/substorage',
+ )
+ );
+
+ $mount = new MountPoint($wrapper2, $this->user . '/files/substorage');
+ Filesystem::getMountManager()->addMount($mount);
+
+ $this->userView->file_put_contents('substorage/subfile.txt', 'foo');
+ $storage2->getScanner()->scan('');
+ $this->assertTrue($storage2->file_exists('subfile.txt'));
+ $this->userView->unlink('substorage/subfile.txt');
+
+ $storage2->getScanner()->scan('');
+ $this->assertFalse($this->userView->getFileInfo('substorage/subfile.txt'));
+ $this->assertFalse($storage2->file_exists('subfile.txt'));
+
+ // check if file is in trashbin
+ $rootView = new \OC\Files\View('/');
+ $results = $rootView->getDirectoryContent($this->user . '/files_trashbin/files');
+ $this->assertEquals(1, count($results));
+ $name = $results[0]->getName();
+ $this->assertEquals('subfile.txt', substr($name, 0, strrpos($name, '.')));
+ }
+}