summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBjoern Schiessle <schiessle@owncloud.com>2015-06-24 15:11:02 +0200
committerBjoern Schiessle <schiessle@owncloud.com>2015-06-24 19:09:20 +0200
commit407db913de26d090567529546be2bab6424607cf (patch)
tree3911442c757cbfc0f18aa5f6f843c901edb84825
parent7a3cfbc3c8d7e2453e962ace592212f00148176e (diff)
downloadnextcloud-server-407db913de26d090567529546be2bab6424607cf.tar.gz
nextcloud-server-407db913de26d090567529546be2bab6424607cf.zip
add occ command to trashbin to remove deleted files
-rw-r--r--apps/files_trashbin/appinfo/register_command.php29
-rw-r--r--apps/files_trashbin/command/cleanup.php118
-rw-r--r--apps/files_trashbin/tests/command/cleanuptest.php192
3 files changed, 339 insertions, 0 deletions
diff --git a/apps/files_trashbin/appinfo/register_command.php b/apps/files_trashbin/appinfo/register_command.php
new file mode 100644
index 00000000000..ae4196dd506
--- /dev/null
+++ b/apps/files_trashbin/appinfo/register_command.php
@@ -0,0 +1,29 @@
+<?php
+/**
+ * @author Björn Schießle <schiessle@owncloud.com>
+ *
+ * @copyright Copyright (c) 2015, ownCloud, Inc.
+ * @license AGPL-3.0
+ *
+ * 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.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Affero General Public License for more details.
+ *
+ * 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/>
+ *
+ */
+
+
+use OCA\Files_Trashbin\Command\CleanUp;
+
+$userManager = OC::$server->getUserManager();
+$rootFolder = \OC::$server->getRootFolder();
+$dbConnection = \OC::$server->getDatabaseConnection();
+/** @var Symfony\Component\Console\Application $application */
+$application->add(new CleanUp($rootFolder, $userManager, $dbConnection));
diff --git a/apps/files_trashbin/command/cleanup.php b/apps/files_trashbin/command/cleanup.php
new file mode 100644
index 00000000000..961f72763ea
--- /dev/null
+++ b/apps/files_trashbin/command/cleanup.php
@@ -0,0 +1,118 @@
+<?php
+/**
+ * @author Björn Schießle <schiessle@owncloud.com>
+ *
+ * @copyright Copyright (c) 2015, ownCloud, Inc.
+ * @license AGPL-3.0
+ *
+ * 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.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Affero General Public License for more details.
+ *
+ * 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/>
+ *
+ */
+
+namespace OCA\Files_Trashbin\Command;
+
+use OCP\Files\IRootFolder;
+use OCP\IDBConnection;
+use OCP\IUserBackend;
+use OCP\IUserManager;
+use Symfony\Component\Console\Command\Command;
+use Symfony\Component\Console\Input\InputArgument;
+use Symfony\Component\Console\Input\InputInterface;
+use Symfony\Component\Console\Output\OutputInterface;
+
+class CleanUp extends Command {
+
+ /** @var IUserManager */
+ protected $userManager;
+
+ /** @var IRootFolder */
+ protected $rootFolder;
+
+ /** @var \OC\DB\Connection */
+ protected $dbConnection;
+
+ /**
+ * @param IRootFolder $rootFolder
+ * @param IUserManager $userManager
+ * @param IDBConnection $dbConnection
+ */
+ function __construct(IRootFolder $rootFolder, IUserManager $userManager, IDBConnection $dbConnection) {
+ parent::__construct();
+ $this->userManager = $userManager;
+ $this->rootFolder = $rootFolder;
+ $this->dbConnection = $dbConnection;
+ }
+
+ protected function configure() {
+ $this
+ ->setName('trashbin:cleanup')
+ ->setDescription('Remove deleted files')
+ ->addArgument(
+ 'user_id',
+ InputArgument::OPTIONAL | InputArgument::IS_ARRAY,
+ 'remove deleted files of the given user(s), if no user is given all deleted files will be removed'
+ );
+ }
+
+ protected function execute(InputInterface $input, OutputInterface $output) {
+ $users = $input->getArgument('user_id');
+ if (!empty($users)) {
+ foreach ($users as $user) {
+ if ($this->userManager->userExists($user)) {
+ $output->writeln("Remove deleted files of <info>$user</info>");
+ $this->removeDeletedFiles($user);
+ } else {
+ $output->writeln("<error>Unknown user $user</error>");
+ }
+ }
+ } else {
+ $output->writeln('Remove all deleted files');
+ foreach ($this->userManager->getBackends() as $backend) {
+ $name = get_class($backend);
+ if ($backend instanceof IUserBackend) {
+ $name = $backend->getBackendName();
+ }
+ $output->writeln("Remove deleted files for users on backend <info>$name</info>");
+ $limit = 500;
+ $offset = 0;
+ do {
+ $users = $backend->getUsers('', $limit, $offset);
+ foreach ($users as $user) {
+ $output->writeln(" <info>$user</info>");
+ $this->removeDeletedFiles($user);
+ }
+ $offset += $limit;
+ } while (count($users) >= $limit);
+ }
+ }
+ }
+
+ /**
+ * remove deleted files for the given user
+ *
+ * @param string $uid
+ */
+ protected function removeDeletedFiles($uid) {
+ \OC_Util::tearDownFS();
+ \OC_Util::setupFS($uid);
+ if ($this->rootFolder->nodeExists('/' . $uid . '/files_trashbin')) {
+ $this->rootFolder->get('/' . $uid . '/files_trashbin')->delete();
+ $query = $this->dbConnection->createQueryBuilder();
+ $query->delete('`*PREFIX*files_trash`')
+ ->where($query->expr()->eq('`user`', ':uid'))
+ ->setParameter('uid', $uid);
+ $query->execute();
+ }
+ }
+
+}
diff --git a/apps/files_trashbin/tests/command/cleanuptest.php b/apps/files_trashbin/tests/command/cleanuptest.php
new file mode 100644
index 00000000000..6e2f78093e0
--- /dev/null
+++ b/apps/files_trashbin/tests/command/cleanuptest.php
@@ -0,0 +1,192 @@
+<?php
+/**
+ * @author Björn Schießle <schiessle@owncloud.com>
+ *
+ * @copyright Copyright (c) 2015, ownCloud, Inc.
+ * @license AGPL-3.0
+ *
+ * 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.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Affero General Public License for more details.
+ *
+ * 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/>
+ *
+ */
+
+
+namespace OCA\Files_Trashbin\Tests\Command;
+
+
+use OCA\Files_Trashbin\Command\CleanUp;
+use Test\TestCase;
+use OC\User\Manager;
+use OCP\Files\IRootFolder;
+
+class CleanUpTest extends TestCase {
+
+ /** @var CleanUp */
+ protected $cleanup;
+
+ /** @var \PHPUnit_Framework_MockObject_MockObject | Manager */
+ protected $userManager;
+
+ /** @var \PHPUnit_Framework_MockObject_MockObject | IRootFolder */
+ protected $rootFolder;
+
+ /** @var \OC\DB\Connection */
+ protected $dbConnection;
+
+ /** @var string */
+ protected $trashTable = '`*PREFIX*files_trash`';
+
+ /** @var string */
+ protected $user0 = 'user0';
+
+ public function setUp() {
+ parent::setUp();
+ $this->rootFolder = $this->getMockBuilder('OCP\Files\IRootFolder')
+ ->disableOriginalConstructor()->getMock();
+ $this->userManager = $this->getMockBuilder('OC\User\Manager')
+ ->disableOriginalConstructor()->getMock();
+
+ $this->dbConnection = \OC::$server->getDatabaseConnection();
+
+ $this->cleanup = new CleanUp($this->rootFolder, $this->userManager, $this->dbConnection);
+ }
+
+ /**
+ * populate files_trash table with 10 dummy values
+ */
+ public function initTable() {
+ $query = $this->dbConnection->createQueryBuilder();
+ $query->delete($this->trashTable)->execute();
+ for ($i = 0; $i < 10; $i++) {
+ $query->insert($this->trashTable)
+ ->values(array(
+ '`id`' => $query->expr()->literal('file'.$i),
+ '`timestamp`' => $query->expr()->literal($i),
+ '`location`' => $query->expr()->literal('.'),
+ '`user`' => $query->expr()->literal('user'.$i%2)
+ ))->execute();
+ }
+ $getAllQuery = $this->dbConnection->createQueryBuilder();
+ $result = $getAllQuery->select('`id`')->from($this->trashTable)->execute()->fetchAll();
+ $this->assertSame(10, count($result));
+ }
+
+ /**
+ * @dataProvider dataTestRemoveDeletedFiles
+ * @param boolean $nodeExists
+ */
+ public function testRemoveDeletedFiles($nodeExists) {
+ $this->initTable();
+ $this->rootFolder->expects($this->once())
+ ->method('nodeExists')
+ ->with('/' . $this->user0 . '/files_trashbin')
+ ->willReturn($nodeExists);
+ if($nodeExists) {
+ $this->rootFolder->expects($this->once())
+ ->method('get')
+ ->with('/' . $this->user0 . '/files_trashbin')
+ ->willReturn($this->rootFolder);
+ $this->rootFolder->expects($this->once())
+ ->method('delete');
+ } else {
+ $this->rootFolder->expects($this->never())->method('get');
+ $this->rootFolder->expects($this->never())->method('delete');
+ }
+ $this->invokePrivate($this->cleanup, 'removeDeletedFiles', [$this->user0]);
+
+ if ($nodeExists) {
+ // if the delete operation was execute only files from user1
+ // should be left.
+ $query = $this->dbConnection->createQueryBuilder();
+ $result = $query->select('`user`')
+ ->from($this->trashTable)
+ ->execute()->fetchAll();
+ $this->assertSame(5, count($result));
+ foreach ($result as $r) {
+ $this->assertSame('user1', $r['user']);
+ }
+ } else {
+ // if no delete operation was execute we should still have all 10
+ // database entries
+ $getAllQuery = $this->dbConnection->createQueryBuilder();
+ $result = $getAllQuery->select('`id`')->from($this->trashTable)->execute()->fetchAll();
+ $this->assertSame(10, count($result));
+ }
+
+ }
+ public function dataTestRemoveDeletedFiles() {
+ return array(
+ array(true),
+ array(false)
+ );
+ }
+
+ /**
+ * test remove deleted files from users given as parameter
+ */
+ public function testExecuteDeleteListOfUsers() {
+ $userIds = ['user1', 'user2', 'user3'];
+ $instance = $this->getMockBuilder('OCA\Files_Trashbin\Command\CleanUp')
+ ->setMethods(['removeDeletedFiles'])
+ ->setConstructorArgs([$this->rootFolder, $this->userManager, $this->dbConnection])
+ ->getMock();
+ $instance->expects($this->exactly(count($userIds)))
+ ->method('removeDeletedFiles')
+ ->willReturnCallback(function ($user) use ($userIds) {
+ $this->assertTrue(in_array($user, $userIds));
+ });
+ $this->userManager->expects($this->exactly(count($userIds)))
+ ->method('userExists')->willReturn(true);
+ $inputInterface = $this->getMockBuilder('\Symfony\Component\Console\Input\InputInterface')
+ ->disableOriginalConstructor()->getMock();
+ $inputInterface->expects($this->once())->method('getArgument')
+ ->with('user_id')
+ ->willReturn($userIds);
+ $outputInterface = $this->getMockBuilder('\Symfony\Component\Console\Output\OutputInterface')
+ ->disableOriginalConstructor()->getMock();
+ $this->invokePrivate($instance, 'execute', [$inputInterface, $outputInterface]);
+ }
+
+ /**
+ * test remove deleted files of all users
+ */
+ public function testExecuteAllUsers() {
+ $userIds = [];
+ $backendUsers = ['user1', 'user2'];
+ $instance = $this->getMockBuilder('OCA\Files_Trashbin\Command\CleanUp')
+ ->setMethods(['removeDeletedFiles'])
+ ->setConstructorArgs([$this->rootFolder, $this->userManager, $this->dbConnection])
+ ->getMock();
+ $backend = $this->getMockBuilder('OC_User_Interface')
+ ->disableOriginalConstructor()->getMock();
+ $backend->expects($this->once())->method('getUsers')
+ ->with('', 500, 0)
+ ->willReturn($backendUsers);
+ $instance->expects($this->exactly(count($backendUsers)))
+ ->method('removeDeletedFiles')
+ ->willReturnCallback(function ($user) use ($backendUsers) {
+ $this->assertTrue(in_array($user, $backendUsers));
+ });
+ $inputInterface = $this->getMockBuilder('\Symfony\Component\Console\Input\InputInterface')
+ ->disableOriginalConstructor()->getMock();
+ $inputInterface->expects($this->once())->method('getArgument')
+ ->with('user_id')
+ ->willReturn($userIds);
+ $outputInterface = $this->getMockBuilder('\Symfony\Component\Console\Output\OutputInterface')
+ ->disableOriginalConstructor()->getMock();
+ $this->userManager->expects($this->once())
+ ->method('getBackends')
+ ->willReturn([$backend]);
+ $this->invokePrivate($instance, 'execute', [$inputInterface, $outputInterface]);
+ }
+
+}