2014-06-17 11:51:49 +00:00
|
|
|
<?php
|
|
|
|
/**
|
2015-03-26 10:44:34 +00:00
|
|
|
* @author Björn Schießle <schiessle@owncloud.com>
|
2015-06-25 09:43:55 +00:00
|
|
|
* @author Clark Tomlinson <fallen013@gmail.com>
|
2015-03-26 10:44:34 +00:00
|
|
|
* @author Joas Schilling <nickvergessen@owncloud.com>
|
|
|
|
* @author Morris Jobke <hey@morrisjobke.de>
|
|
|
|
* @author Robin Appelman <icewind@owncloud.com>
|
2016-01-12 14:02:16 +00:00
|
|
|
* @author Roeland Jago Douma <rullzer@owncloud.com>
|
|
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
2015-03-26 10:44:34 +00:00
|
|
|
* @author Vincent Petry <pvince81@owncloud.com>
|
2014-06-17 11:51:49 +00:00
|
|
|
*
|
2016-01-12 14:02:16 +00:00
|
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
2015-03-26 10:44:34 +00:00
|
|
|
* @license AGPL-3.0
|
2014-06-17 11:51:49 +00:00
|
|
|
*
|
2015-03-26 10:44:34 +00:00
|
|
|
* This code is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
|
|
* as published by the Free Software Foundation.
|
2014-06-17 11:51:49 +00:00
|
|
|
*
|
2015-03-26 10:44:34 +00:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
2014-06-17 11:51:49 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2015-03-26 10:44:34 +00:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
2014-06-17 11:51:49 +00:00
|
|
|
*
|
2015-03-26 10:44:34 +00:00
|
|
|
* You should have received a copy of the GNU Affero General Public License, version 3,
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
2014-06-17 11:51:49 +00:00
|
|
|
*
|
|
|
|
*/
|
2015-02-26 10:37:37 +00:00
|
|
|
|
2014-06-17 11:51:49 +00:00
|
|
|
use OCA\Files_Trashbin;
|
|
|
|
|
|
|
|
/**
|
2015-04-02 11:28:18 +00:00
|
|
|
* Class Test_Encryption
|
2015-11-20 10:27:11 +00:00
|
|
|
*
|
|
|
|
* @group DB
|
2014-06-17 11:51:49 +00:00
|
|
|
*/
|
2014-11-10 21:28:12 +00:00
|
|
|
class Test_Trashbin extends \Test\TestCase {
|
2014-06-17 11:51:49 +00:00
|
|
|
|
|
|
|
const TEST_TRASHBIN_USER1 = "test-trashbin-user1";
|
2014-06-17 20:30:11 +00:00
|
|
|
const TEST_TRASHBIN_USER2 = "test-trashbin-user2";
|
2014-06-17 11:51:49 +00:00
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
private $trashRoot1;
|
|
|
|
private $trashRoot2;
|
|
|
|
|
|
|
|
private static $rememberRetentionObligation;
|
2014-06-17 11:51:49 +00:00
|
|
|
|
2015-05-12 11:14:57 +00:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private static $trashBinStatus;
|
|
|
|
|
2014-06-17 11:51:49 +00:00
|
|
|
/**
|
|
|
|
* @var \OC\Files\View
|
|
|
|
*/
|
|
|
|
private $rootView;
|
|
|
|
|
|
|
|
public static function setUpBeforeClass() {
|
2014-11-10 21:28:12 +00:00
|
|
|
parent::setUpBeforeClass();
|
|
|
|
|
2015-05-12 11:14:57 +00:00
|
|
|
$appManager = \OC::$server->getAppManager();
|
|
|
|
self::$trashBinStatus = $appManager->isEnabledForUser('files_trashbin');
|
|
|
|
|
2014-06-17 11:51:49 +00:00
|
|
|
// reset backend
|
|
|
|
\OC_User::clearBackends();
|
|
|
|
\OC_User::useBackend('database');
|
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
// clear share hooks
|
|
|
|
\OC_Hook::clear('OCP\\Share');
|
|
|
|
\OC::registerShareHooks();
|
2015-03-27 12:44:40 +00:00
|
|
|
$application = new \OCA\Files_Sharing\AppInfo\Application();
|
|
|
|
$application->registerMountProviders();
|
2014-06-17 20:30:11 +00:00
|
|
|
|
|
|
|
//disable encryption
|
2015-04-30 17:02:27 +00:00
|
|
|
\OC_App::disable('encryption');
|
2014-06-17 20:30:11 +00:00
|
|
|
|
2015-12-02 14:56:59 +00:00
|
|
|
$config = \OC::$server->getConfig();
|
2014-06-17 20:30:11 +00:00
|
|
|
//configure trashbin
|
2015-12-02 14:56:59 +00:00
|
|
|
self::$rememberRetentionObligation = $config->getSystemValue('trashbin_retention_obligation', Files_Trashbin\Expiration::DEFAULT_RETENTION_OBLIGATION);
|
|
|
|
$config->setSystemValue('trashbin_retention_obligation', 'auto, 2');
|
2014-06-17 20:30:11 +00:00
|
|
|
|
2014-06-17 11:51:49 +00:00
|
|
|
// register hooks
|
|
|
|
Files_Trashbin\Trashbin::registerHooks();
|
|
|
|
|
|
|
|
// create test user
|
2014-06-17 20:30:11 +00:00
|
|
|
self::loginHelper(self::TEST_TRASHBIN_USER2, true);
|
2014-06-17 11:51:49 +00:00
|
|
|
self::loginHelper(self::TEST_TRASHBIN_USER1, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public static function tearDownAfterClass() {
|
|
|
|
// cleanup test user
|
2015-12-17 14:59:23 +00:00
|
|
|
$user = \OC::$server->getUserManager()->get(self::TEST_TRASHBIN_USER1);
|
|
|
|
if ($user !== null) { $user->delete(); }
|
2014-06-17 11:51:49 +00:00
|
|
|
|
2015-12-02 14:56:59 +00:00
|
|
|
\OC::$server->getConfig()->setSystemValue('trashbin_retention_obligation', self::$rememberRetentionObligation);
|
2014-06-17 20:30:11 +00:00
|
|
|
|
2014-06-17 11:51:49 +00:00
|
|
|
\OC_Hook::clear();
|
2014-11-10 21:28:12 +00:00
|
|
|
|
2015-01-23 13:55:38 +00:00
|
|
|
\OC\Files\Filesystem::getLoader()->removeStorageWrapper('oc_trashbin');
|
|
|
|
|
2015-05-12 11:14:57 +00:00
|
|
|
if (self::$trashBinStatus) {
|
|
|
|
\OC::$server->getAppManager()->enableApp('files_trashbin');
|
|
|
|
}
|
|
|
|
|
2014-11-10 21:28:12 +00:00
|
|
|
parent::tearDownAfterClass();
|
2014-06-17 11:51:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 21:28:12 +00:00
|
|
|
protected function setUp() {
|
|
|
|
parent::setUp();
|
|
|
|
|
2015-05-12 11:14:57 +00:00
|
|
|
\OC::$server->getAppManager()->enableApp('files_trashbin');
|
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
$this->trashRoot1 = '/' . self::TEST_TRASHBIN_USER1 . '/files_trashbin';
|
|
|
|
$this->trashRoot2 = '/' . self::TEST_TRASHBIN_USER2 . '/files_trashbin';
|
2014-06-17 11:51:49 +00:00
|
|
|
$this->rootView = new \OC\Files\View();
|
2014-06-17 20:30:11 +00:00
|
|
|
self::loginHelper(self::TEST_TRASHBIN_USER1);
|
2014-06-17 11:51:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 21:28:12 +00:00
|
|
|
protected function tearDown() {
|
2015-05-12 11:14:57 +00:00
|
|
|
// disable trashbin to be able to properly clean up
|
|
|
|
\OC::$server->getAppManager()->disableApp('files_trashbin');
|
|
|
|
|
|
|
|
$this->rootView->deleteAll('/' . self::TEST_TRASHBIN_USER1 . '/files');
|
|
|
|
$this->rootView->deleteAll('/' . self::TEST_TRASHBIN_USER2 . '/files');
|
2014-06-17 20:30:11 +00:00
|
|
|
$this->rootView->deleteAll($this->trashRoot1);
|
|
|
|
$this->rootView->deleteAll($this->trashRoot2);
|
2014-11-10 21:28:12 +00:00
|
|
|
|
2015-05-12 11:14:57 +00:00
|
|
|
// clear trash table
|
|
|
|
$connection = \OC::$server->getDatabaseConnection();
|
|
|
|
$connection->executeUpdate('DELETE FROM `*PREFIX*files_trash`');
|
|
|
|
|
2014-11-10 21:28:12 +00:00
|
|
|
parent::tearDown();
|
2014-06-17 11:51:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* test expiration of files older then the max storage time defined for the trash
|
|
|
|
*/
|
|
|
|
public function testExpireOldFiles() {
|
|
|
|
|
|
|
|
$currentTime = time();
|
|
|
|
$expireAt = $currentTime - 2*24*60*60;
|
|
|
|
$expiredDate = $currentTime - 3*24*60*60;
|
|
|
|
|
|
|
|
// create some files
|
|
|
|
\OC\Files\Filesystem::file_put_contents('file1.txt', 'file1');
|
|
|
|
\OC\Files\Filesystem::file_put_contents('file2.txt', 'file2');
|
|
|
|
\OC\Files\Filesystem::file_put_contents('file3.txt', 'file3');
|
|
|
|
|
|
|
|
// delete them so that they end up in the trash bin
|
|
|
|
\OC\Files\Filesystem::unlink('file1.txt');
|
|
|
|
\OC\Files\Filesystem::unlink('file2.txt');
|
|
|
|
\OC\Files\Filesystem::unlink('file3.txt');
|
|
|
|
|
|
|
|
//make sure that files are in the trash bin
|
2014-06-17 20:30:11 +00:00
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'name');
|
2014-06-17 11:51:49 +00:00
|
|
|
$this->assertSame(3, count($filesInTrash));
|
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
// every second file will get a date in the past so that it will get expired
|
|
|
|
$manipulatedList = $this->manipulateDeleteTime($filesInTrash, $this->trashRoot1, $expiredDate);
|
2014-06-17 11:51:49 +00:00
|
|
|
|
|
|
|
$testClass = new TrashbinForTesting();
|
|
|
|
list($sizeOfDeletedFiles, $count) = $testClass->dummyDeleteExpiredFiles($manipulatedList, $expireAt);
|
|
|
|
|
|
|
|
$this->assertSame(10, $sizeOfDeletedFiles);
|
|
|
|
$this->assertSame(2, $count);
|
|
|
|
|
|
|
|
// only file2.txt should be left
|
|
|
|
$remainingFiles = array_slice($manipulatedList, $count);
|
|
|
|
$this->assertSame(1, count($remainingFiles));
|
|
|
|
$remainingFile = reset($remainingFiles);
|
|
|
|
$this->assertSame('file2.txt', $remainingFile['name']);
|
|
|
|
|
|
|
|
// check that file1.txt and file3.txt was really deleted
|
2014-06-17 18:08:40 +00:00
|
|
|
$newTrashContent = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1);
|
2014-06-17 11:51:49 +00:00
|
|
|
$this->assertSame(1, count($newTrashContent));
|
|
|
|
$element = reset($newTrashContent);
|
|
|
|
$this->assertSame('file2.txt', $element['name']);
|
|
|
|
}
|
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
/**
|
|
|
|
* test expiration of files older then the max storage time defined for the trash
|
|
|
|
* in this test we delete a shared file and check if both trash bins, the one from
|
|
|
|
* the owner of the file and the one from the user who deleted the file get expired
|
|
|
|
* correctly
|
|
|
|
*/
|
|
|
|
public function testExpireOldFilesShared() {
|
|
|
|
|
|
|
|
$currentTime = time();
|
|
|
|
$folder = "trashTest-" . $currentTime . '/';
|
|
|
|
$expiredDate = $currentTime - 3*24*60*60;
|
|
|
|
|
|
|
|
// create some files
|
|
|
|
\OC\Files\Filesystem::mkdir($folder);
|
|
|
|
\OC\Files\Filesystem::file_put_contents($folder . 'user1-1.txt', 'file1');
|
|
|
|
\OC\Files\Filesystem::file_put_contents($folder . 'user1-2.txt', 'file2');
|
|
|
|
\OC\Files\Filesystem::file_put_contents($folder . 'user1-3.txt', 'file3');
|
|
|
|
\OC\Files\Filesystem::file_put_contents($folder . 'user1-4.txt', 'file4');
|
|
|
|
|
|
|
|
//share user1-4.txt with user2
|
|
|
|
$fileInfo = \OC\Files\Filesystem::getFileInfo($folder);
|
|
|
|
$result = \OCP\Share::shareItem('folder', $fileInfo->getId(), \OCP\Share::SHARE_TYPE_USER, self::TEST_TRASHBIN_USER2, 31);
|
|
|
|
$this->assertTrue($result);
|
|
|
|
|
|
|
|
// delete them so that they end up in the trash bin
|
|
|
|
\OC\Files\Filesystem::unlink($folder . 'user1-1.txt');
|
|
|
|
\OC\Files\Filesystem::unlink($folder . 'user1-2.txt');
|
|
|
|
\OC\Files\Filesystem::unlink($folder . 'user1-3.txt');
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'name');
|
|
|
|
$this->assertSame(3, count($filesInTrash));
|
|
|
|
|
|
|
|
// every second file will get a date in the past so that it will get expired
|
|
|
|
$this->manipulateDeleteTime($filesInTrash, $this->trashRoot1, $expiredDate);
|
|
|
|
|
|
|
|
// login as user2
|
|
|
|
self::loginHelper(self::TEST_TRASHBIN_USER2);
|
|
|
|
|
|
|
|
$this->assertTrue(\OC\Files\Filesystem::file_exists($folder . "user1-4.txt"));
|
|
|
|
|
|
|
|
// create some files
|
|
|
|
\OC\Files\Filesystem::file_put_contents('user2-1.txt', 'file1');
|
|
|
|
\OC\Files\Filesystem::file_put_contents('user2-2.txt', 'file2');
|
|
|
|
|
|
|
|
// delete them so that they end up in the trash bin
|
|
|
|
\OC\Files\Filesystem::unlink('user2-1.txt');
|
|
|
|
\OC\Files\Filesystem::unlink('user2-2.txt');
|
|
|
|
|
|
|
|
$filesInTrashUser2 = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER2, 'name');
|
|
|
|
$this->assertSame(2, count($filesInTrashUser2));
|
|
|
|
|
|
|
|
// every second file will get a date in the past so that it will get expired
|
|
|
|
$this->manipulateDeleteTime($filesInTrashUser2, $this->trashRoot2, $expiredDate);
|
|
|
|
|
|
|
|
\OC\Files\Filesystem::unlink($folder . 'user1-4.txt');
|
|
|
|
|
2015-03-02 14:25:50 +00:00
|
|
|
$this->runCommands();
|
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
$filesInTrashUser2AfterDelete = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER2);
|
|
|
|
|
|
|
|
// user2-1.txt should have been expired
|
|
|
|
$this->verifyArray($filesInTrashUser2AfterDelete, array('user2-2.txt', 'user1-4.txt'));
|
|
|
|
|
2015-10-15 13:47:13 +00:00
|
|
|
self::loginHelper(self::TEST_TRASHBIN_USER1);
|
|
|
|
|
2014-06-17 20:30:11 +00:00
|
|
|
// user1-1.txt and user1-3.txt should have been expired
|
|
|
|
$filesInTrashUser1AfterDelete = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1);
|
|
|
|
|
|
|
|
$this->verifyArray($filesInTrashUser1AfterDelete, array('user1-2.txt', 'user1-4.txt'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* verify that the array contains the expected results
|
2015-12-07 20:08:20 +00:00
|
|
|
* @param OCP\Files\FileInfo[] $result
|
|
|
|
* @param string[] $expected
|
2014-06-17 20:30:11 +00:00
|
|
|
*/
|
|
|
|
private function verifyArray($result, $expected) {
|
|
|
|
$this->assertSame(count($expected), count($result));
|
|
|
|
foreach ($expected as $expectedFile) {
|
|
|
|
$found = false;
|
|
|
|
foreach ($result as $fileInTrash) {
|
|
|
|
if ($expectedFile === $fileInTrash['name']) {
|
|
|
|
$found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!$found) {
|
|
|
|
// if we didn't found the expected file, something went wrong
|
|
|
|
$this->assertTrue(false, "can't find expected file '" . $expectedFile . "' in trash bin");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-07 20:08:20 +00:00
|
|
|
/**
|
|
|
|
* @param OCP\Files\FileInfo[] $files
|
|
|
|
* @param string $trashRoot
|
|
|
|
* @param integer $expireDate
|
|
|
|
*/
|
2014-06-17 20:30:11 +00:00
|
|
|
private function manipulateDeleteTime($files, $trashRoot, $expireDate) {
|
2014-06-17 11:51:49 +00:00
|
|
|
$counter = 0;
|
|
|
|
foreach ($files as &$file) {
|
|
|
|
// modify every second file
|
|
|
|
$counter = ($counter + 1) % 2;
|
|
|
|
if ($counter === 1) {
|
2014-06-17 20:30:11 +00:00
|
|
|
$source = $trashRoot . '/files/' . $file['name'].'.d'.$file['mtime'];
|
|
|
|
$target = \OC\Files\Filesystem::normalizePath($trashRoot . '/files/' . $file['name'] . '.d' . $expireDate);
|
2014-06-17 11:51:49 +00:00
|
|
|
$this->rootView->rename($source, $target);
|
|
|
|
$file['mtime'] = $expireDate;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return \OCA\Files\Helper::sortFiles($files, 'mtime');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* test expiration of old files in the trash bin until the max size
|
|
|
|
* of the trash bin is met again
|
|
|
|
*/
|
|
|
|
public function testExpireOldFilesUtilLimitsAreMet() {
|
|
|
|
|
|
|
|
// create some files
|
|
|
|
\OC\Files\Filesystem::file_put_contents('file1.txt', 'file1');
|
|
|
|
\OC\Files\Filesystem::file_put_contents('file2.txt', 'file2');
|
|
|
|
\OC\Files\Filesystem::file_put_contents('file3.txt', 'file3');
|
|
|
|
|
|
|
|
// delete them so that they end up in the trash bin
|
|
|
|
\OC\Files\Filesystem::unlink('file3.txt');
|
|
|
|
sleep(1); // make sure that every file has a unique mtime
|
|
|
|
\OC\Files\Filesystem::unlink('file2.txt');
|
|
|
|
sleep(1); // make sure that every file has a unique mtime
|
|
|
|
\OC\Files\Filesystem::unlink('file1.txt');
|
|
|
|
|
|
|
|
//make sure that files are in the trash bin
|
2014-06-17 18:08:40 +00:00
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
2014-06-17 11:51:49 +00:00
|
|
|
$this->assertSame(3, count($filesInTrash));
|
|
|
|
|
|
|
|
$testClass = new TrashbinForTesting();
|
|
|
|
$sizeOfDeletedFiles = $testClass->dummyDeleteFiles($filesInTrash, -8);
|
|
|
|
|
|
|
|
// the two oldest files (file3.txt and file2.txt) should be deleted
|
|
|
|
$this->assertSame(10, $sizeOfDeletedFiles);
|
|
|
|
|
2014-06-17 18:08:40 +00:00
|
|
|
$newTrashContent = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1);
|
2014-06-17 11:51:49 +00:00
|
|
|
$this->assertSame(1, count($newTrashContent));
|
|
|
|
$element = reset($newTrashContent);
|
|
|
|
$this->assertSame('file1.txt', $element['name']);
|
|
|
|
}
|
|
|
|
|
2015-05-12 11:14:57 +00:00
|
|
|
/**
|
|
|
|
* Test restoring a file
|
|
|
|
*/
|
|
|
|
public function testRestoreFileInRoot() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$file = $userFolder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('file1.txt'));
|
|
|
|
|
|
|
|
$file->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('file1.txt'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'file1.txt.d' . $trashedFile->getMtime(),
|
|
|
|
$trashedFile->getName(),
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$file = $userFolder->get('file1.txt');
|
|
|
|
$this->assertEquals('foo', $file->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a file in subfolder
|
|
|
|
*/
|
|
|
|
public function testRestoreFileInSubfolder() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$folder = $userFolder->newFolder('folder');
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$file->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'file1.txt.d' . $trashedFile->getMtime(),
|
|
|
|
$trashedFile->getName(),
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$file = $userFolder->get('folder/file1.txt');
|
|
|
|
$this->assertEquals('foo', $file->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a folder
|
|
|
|
*/
|
|
|
|
public function testRestoreFolder() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$folder = $userFolder->newFolder('folder');
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('folder'));
|
|
|
|
|
|
|
|
$folder->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('folder'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFolder = $filesInTrash[0];
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'folder.d' . $trashedFolder->getMtime(),
|
|
|
|
$trashedFolder->getName(),
|
|
|
|
$trashedFolder->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$file = $userFolder->get('folder/file1.txt');
|
|
|
|
$this->assertEquals('foo', $file->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a file from inside a trashed folder
|
|
|
|
*/
|
|
|
|
public function testRestoreFileFromTrashedSubfolder() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$folder = $userFolder->newFolder('folder');
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('folder'));
|
|
|
|
|
|
|
|
$folder->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('folder'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'folder.d' . $trashedFile->getMtime() . '/file1.txt',
|
|
|
|
'file1.txt',
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$file = $userFolder->get('file1.txt');
|
|
|
|
$this->assertEquals('foo', $file->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a file whenever the source folder was removed.
|
|
|
|
* The file should then land in the root.
|
|
|
|
*/
|
|
|
|
public function testRestoreFileWithMissingSourceFolder() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$folder = $userFolder->newFolder('folder');
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$file->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
// delete source folder
|
|
|
|
$folder->delete();
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'file1.txt.d' . $trashedFile->getMtime(),
|
|
|
|
$trashedFile->getName(),
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$file = $userFolder->get('file1.txt');
|
|
|
|
$this->assertEquals('foo', $file->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a file in the root folder whenever there is another file
|
|
|
|
* with the same name in the root folder
|
|
|
|
*/
|
|
|
|
public function testRestoreFileDoesNotOverwriteExistingInRoot() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$file = $userFolder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('file1.txt'));
|
|
|
|
|
|
|
|
$file->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('file1.txt'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
// create another file
|
|
|
|
$file = $userFolder->newFile('file1.txt');
|
|
|
|
$file->putContent('bar');
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'file1.txt.d' . $trashedFile->getMtime(),
|
|
|
|
$trashedFile->getName(),
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$anotherFile = $userFolder->get('file1.txt');
|
|
|
|
$this->assertEquals('bar', $anotherFile->getContent());
|
|
|
|
|
|
|
|
$restoredFile = $userFolder->get('file1 (restored).txt');
|
|
|
|
$this->assertEquals('foo', $restoredFile->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a file whenever there is another file
|
|
|
|
* with the same name in the source folder
|
|
|
|
*/
|
|
|
|
public function testRestoreFileDoesNotOverwriteExistingInSubfolder() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$folder = $userFolder->newFolder('folder');
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$file->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
// create another file
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('bar');
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'file1.txt.d' . $trashedFile->getMtime(),
|
|
|
|
$trashedFile->getName(),
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$anotherFile = $userFolder->get('folder/file1.txt');
|
|
|
|
$this->assertEquals('bar', $anotherFile->getContent());
|
|
|
|
|
|
|
|
$restoredFile = $userFolder->get('folder/file1 (restored).txt');
|
|
|
|
$this->assertEquals('foo', $restoredFile->getContent());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a non-existing file from trashbin, returns false
|
|
|
|
*/
|
|
|
|
public function testRestoreUnexistingFile() {
|
|
|
|
$this->assertFalse(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'unexist.txt.d123456',
|
|
|
|
'unexist.txt',
|
|
|
|
'123456'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test restoring a file into a read-only folder, will restore
|
|
|
|
* the file to root instead
|
|
|
|
*/
|
|
|
|
public function testRestoreFileIntoReadOnlySourceFolder() {
|
|
|
|
$userFolder = \OC::$server->getUserFolder();
|
|
|
|
$folder = $userFolder->newFolder('folder');
|
|
|
|
$file = $folder->newFile('file1.txt');
|
|
|
|
$file->putContent('foo');
|
|
|
|
|
|
|
|
$this->assertTrue($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$file->delete();
|
|
|
|
|
|
|
|
$this->assertFalse($userFolder->nodeExists('folder/file1.txt'));
|
|
|
|
|
|
|
|
$filesInTrash = OCA\Files_Trashbin\Helper::getTrashFiles('/', self::TEST_TRASHBIN_USER1, 'mtime');
|
|
|
|
$this->assertCount(1, $filesInTrash);
|
|
|
|
|
|
|
|
/** @var \OCP\Files\FileInfo */
|
|
|
|
$trashedFile = $filesInTrash[0];
|
|
|
|
|
|
|
|
// delete source folder
|
|
|
|
list($storage, $internalPath) = $this->rootView->resolvePath('/' . self::TEST_TRASHBIN_USER1 . '/files/folder');
|
2015-10-13 10:19:07 +00:00
|
|
|
if ($storage instanceof \OC\Files\Storage\Local) {
|
|
|
|
$folderAbsPath = $storage->getSourcePath($internalPath);
|
|
|
|
// make folder read-only
|
|
|
|
chmod($folderAbsPath, 0555);
|
2015-05-12 11:14:57 +00:00
|
|
|
|
2015-10-13 10:19:07 +00:00
|
|
|
$this->assertTrue(
|
|
|
|
OCA\Files_Trashbin\Trashbin::restore(
|
|
|
|
'file1.txt.d' . $trashedFile->getMtime(),
|
|
|
|
$trashedFile->getName(),
|
|
|
|
$trashedFile->getMtime()
|
|
|
|
)
|
|
|
|
);
|
2015-05-12 11:14:57 +00:00
|
|
|
|
2015-10-13 10:19:07 +00:00
|
|
|
$file = $userFolder->get('file1.txt');
|
|
|
|
$this->assertEquals('foo', $file->getContent());
|
2015-05-12 11:14:57 +00:00
|
|
|
|
2015-10-13 10:19:07 +00:00
|
|
|
chmod($folderAbsPath, 0755);
|
|
|
|
}
|
2015-05-12 11:14:57 +00:00
|
|
|
}
|
|
|
|
|
2014-06-17 11:51:49 +00:00
|
|
|
/**
|
|
|
|
* @param string $user
|
|
|
|
* @param bool $create
|
|
|
|
*/
|
|
|
|
public static function loginHelper($user, $create = false) {
|
|
|
|
if ($create) {
|
|
|
|
try {
|
2015-12-17 14:10:11 +00:00
|
|
|
\OC::$server->getUserManager()->createUser($user, $user);
|
2014-06-17 11:51:49 +00:00
|
|
|
} catch(\Exception $e) { // catch username is already being used from previous aborted runs
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
\OC_Util::tearDownFS();
|
|
|
|
\OC_User::setUserId('');
|
|
|
|
\OC\Files\Filesystem::tearDown();
|
|
|
|
\OC_User::setUserId($user);
|
|
|
|
\OC_Util::setupFS($user);
|
2015-05-11 14:50:43 +00:00
|
|
|
\OC::$server->getUserFolder($user);
|
2014-06-17 11:51:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// just a dummy class to make protected methods available for testing
|
|
|
|
class TrashbinForTesting extends Files_Trashbin\Trashbin {
|
2015-12-07 20:08:20 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param OCP\Files\FileInfo[] $files
|
|
|
|
* @param integer $limit
|
|
|
|
*/
|
2014-06-17 11:51:49 +00:00
|
|
|
public function dummyDeleteExpiredFiles($files, $limit) {
|
|
|
|
// dummy value for $retention_obligation because it is not needed here
|
2014-06-17 18:08:40 +00:00
|
|
|
return parent::deleteExpiredFiles($files, \Test_Trashbin::TEST_TRASHBIN_USER1, $limit, 0);
|
2014-06-17 11:51:49 +00:00
|
|
|
}
|
|
|
|
|
2015-12-07 20:08:20 +00:00
|
|
|
/**
|
|
|
|
* @param OCP\Files\FileInfo[] $files
|
|
|
|
* @param integer $availableSpace
|
|
|
|
*/
|
2014-06-17 11:51:49 +00:00
|
|
|
public function dummyDeleteFiles($files, $availableSpace) {
|
2014-06-17 18:08:40 +00:00
|
|
|
return parent::deleteFiles($files, \Test_Trashbin::TEST_TRASHBIN_USER1, $availableSpace);
|
2014-06-17 11:51:49 +00:00
|
|
|
}
|
|
|
|
}
|