diff --git a/apps/files_trashbin/appinfo/register_command.php b/apps/files_trashbin/appinfo/register_command.php
new file mode 100644
index 0000000000000000000000000000000000000000..ae4196dd50603c8148102197346adc2b2ef9a257
--- /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 0000000000000000000000000000000000000000..961f72763eaa33e1dddfe59832580e973f00a02e
--- /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 0000000000000000000000000000000000000000..6e2f78093e0cf5388a90709139437c817d0c1acb
--- /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]);
+	}
+
+}