Merge pull request #3044 from owncloud/backgroundwatcher
Add a background job to watch for external filesystem changes
This commit is contained in:
commit
5b61c5d478
4 changed files with 126 additions and 3 deletions
|
@ -18,4 +18,6 @@ OC_Search::registerProvider('OC_Search_Provider_File');
|
|||
\OC_Hook::connect('OC_Filesystem', 'post_write', '\OC\Files\Cache\Updater', 'writeHook');
|
||||
\OC_Hook::connect('OC_Filesystem', 'post_touch', '\OC\Files\Cache\Updater', 'touchHook');
|
||||
\OC_Hook::connect('OC_Filesystem', 'post_delete', '\OC\Files\Cache\Updater', 'deleteHook');
|
||||
\OC_Hook::connect('OC_Filesystem', 'post_rename', '\OC\Files\Cache\Updater', 'renameHook');
|
||||
\OC_Hook::connect('OC_Filesystem', 'post_rename', '\OC\Files\Cache\Updater', 'renameHook');
|
||||
|
||||
\OC_BackgroundJob_RegularTask::register('\OC\Files\Cache\BackgroundWatcher', 'checkNext');
|
||||
|
|
104
lib/files/cache/backgroundwatcher.php
vendored
Normal file
104
lib/files/cache/backgroundwatcher.php
vendored
Normal file
|
@ -0,0 +1,104 @@
|
|||
<?php
|
||||
/**
|
||||
* Copyright (c) 2013 Robin Appelman <icewind@owncloud.com>
|
||||
* This file is licensed under the Affero General Public License version 3 or
|
||||
* later.
|
||||
* See the COPYING-README file.
|
||||
*/
|
||||
|
||||
namespace OC\Files\Cache;
|
||||
|
||||
use \OC\Files\Mount;
|
||||
use \OC\Files\Filesystem;
|
||||
|
||||
class BackgroundWatcher {
|
||||
static $folderMimetype = null;
|
||||
|
||||
static private function getFolderMimetype() {
|
||||
if (!is_null(self::$folderMimetype)) {
|
||||
return self::$folderMimetype;
|
||||
}
|
||||
$query = \OC_DB::prepare('SELECT `id` FROM `*PREFIX*mimetypes` WHERE `mimetype` = ?');
|
||||
$result = $query->execute(array('httpd/unix-directory'));
|
||||
$row = $result->fetchRow();
|
||||
return $row['id'];
|
||||
}
|
||||
|
||||
static private function checkUpdate($id) {
|
||||
$cacheItem = Cache::getById($id);
|
||||
if (is_null($cacheItem)) {
|
||||
return;
|
||||
}
|
||||
list($storageId, $internalPath) = $cacheItem;
|
||||
$mounts = Mount::findByStorageId($storageId);
|
||||
|
||||
if (count($mounts) === 0) {
|
||||
//if the storage we need isn't mounted on default, try to find a user that has access to the storage
|
||||
$permissionsCache = new Permissions($storageId);
|
||||
$users = $permissionsCache->getUsers($id);
|
||||
if (count($users) === 0) {
|
||||
return;
|
||||
}
|
||||
Filesystem::initMountPoints($users[0]);
|
||||
$mounts = Mount::findByStorageId($storageId);
|
||||
if (count($mounts) === 0) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
$storage = $mounts[0]->getStorage();
|
||||
$watcher = new Watcher($storage);
|
||||
$watcher->checkUpdate($internalPath);
|
||||
}
|
||||
|
||||
/**
|
||||
* get the next fileid in the cache
|
||||
*
|
||||
* @param int $previous
|
||||
* @param bool $folder
|
||||
* @return int
|
||||
*/
|
||||
static private function getNextFileId($previous, $folder) {
|
||||
if ($folder) {
|
||||
$query = \OC_DB::prepare('SELECT `fileid` FROM `*PREFIX*filecache` WHERE `fileid` > ? AND mimetype = ' . self::getFolderMimetype() . ' ORDER BY `fileid` ASC', 1);
|
||||
} else {
|
||||
$query = \OC_DB::prepare('SELECT `fileid` FROM `*PREFIX*filecache` WHERE `fileid` > ? AND mimetype != ' . self::getFolderMimetype() . ' ORDER BY `fileid` ASC', 1);
|
||||
}
|
||||
$result = $query->execute(array($previous));
|
||||
if ($row = $result->fetchRow()) {
|
||||
return $row['fileid'];
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
static public function checkNext() {
|
||||
// check both 1 file and 1 folder, this way new files are detected quicker because there are less folders than files usually
|
||||
$previousFile = \OC_Appconfig::getValue('files', 'backgroundwatcher_previous_file', 0);
|
||||
$previousFolder = \OC_Appconfig::getValue('files', 'backgroundwatcher_previous_folder', 0);
|
||||
$nextFile = self::getNextFileId($previousFile, false);
|
||||
$nextFolder = self::getNextFileId($previousFolder, true);
|
||||
\OC_Appconfig::setValue('files', 'backgroundwatcher_previous_file', $nextFile);
|
||||
\OC_Appconfig::setValue('files', 'backgroundwatcher_previous_folder', $nextFolder);
|
||||
if ($nextFile > 0) {
|
||||
self::checkUpdate($nextFile);
|
||||
}
|
||||
if ($nextFolder > 0) {
|
||||
self::checkUpdate($nextFolder);
|
||||
}
|
||||
}
|
||||
|
||||
static public function checkAll() {
|
||||
$previous = 0;
|
||||
$next = 1;
|
||||
while ($next != 0) {
|
||||
$next = self::getNextFileId($previous, true);
|
||||
self::checkUpdate($next);
|
||||
}
|
||||
$previous = 0;
|
||||
$next = 1;
|
||||
while ($next != 0) {
|
||||
$next = self::getNextFileId($previous, false);
|
||||
self::checkUpdate($next);
|
||||
}
|
||||
}
|
||||
}
|
15
lib/files/cache/permissions.php
vendored
15
lib/files/cache/permissions.php
vendored
|
@ -107,4 +107,19 @@ class Permissions {
|
|||
$query->execute(array($fileId, $user));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* get the list of users which have permissions stored for a file
|
||||
*
|
||||
* @param int $fileId
|
||||
*/
|
||||
public function getUsers($fileId) {
|
||||
$query = \OC_DB::prepare('SELECT `user` FROM `*PREFIX*permissions` WHERE `fileid` = ?');
|
||||
$result = $query->execute(array($fileId));
|
||||
$users = array();
|
||||
while ($row = $result->fetchRow()) {
|
||||
$users[] = $row['user'];
|
||||
}
|
||||
return $users;
|
||||
}
|
||||
}
|
||||
|
|
6
tests/lib/files/cache/permissions.php
vendored
6
tests/lib/files/cache/permissions.php
vendored
|
@ -14,8 +14,8 @@ class Permissions extends \PHPUnit_Framework_TestCase {
|
|||
*/
|
||||
private $permissionsCache;
|
||||
|
||||
function setUp(){
|
||||
$this->permissionsCache=new \OC\Files\Cache\Permissions('dummy');
|
||||
function setUp() {
|
||||
$this->permissionsCache = new \OC\Files\Cache\Permissions('dummy');
|
||||
}
|
||||
|
||||
function testSimple() {
|
||||
|
@ -23,8 +23,10 @@ class Permissions extends \PHPUnit_Framework_TestCase {
|
|||
$user = uniqid();
|
||||
|
||||
$this->assertEquals(-1, $this->permissionsCache->get(1, $user));
|
||||
$this->assertNotContains($user, $this->permissionsCache->getUsers(1));
|
||||
$this->permissionsCache->set(1, $user, 1);
|
||||
$this->assertEquals(1, $this->permissionsCache->get(1, $user));
|
||||
$this->assertContains($user, $this->permissionsCache->getUsers(1));
|
||||
$this->assertEquals(-1, $this->permissionsCache->get(2, $user));
|
||||
$this->assertEquals(-1, $this->permissionsCache->get(1, $user . '2'));
|
||||
|
||||
|
|
Loading…
Reference in a new issue