From 5365ae416ec1a2c1dd868707cbb01f50bbe9027d Mon Sep 17 00:00:00 2001 From: ringmaster Date: Tue, 20 May 2014 17:44:57 -0400 Subject: flock changes. Work in progress. --- lib/private/files/storage/local.php | 14 +++ .../files/storage/wrapper/lockingwrapper.php | 139 +++++++++++++++++++++ lib/private/util.php | 10 ++ 3 files changed, 163 insertions(+) create mode 100644 lib/private/files/storage/wrapper/lockingwrapper.php (limited to 'lib/private') diff --git a/lib/private/files/storage/local.php b/lib/private/files/storage/local.php index e33747bbd52..565f9202d5c 100644 --- a/lib/private/files/storage/local.php +++ b/lib/private/files/storage/local.php @@ -279,5 +279,19 @@ if (\OC_Util::runningOnWindows()) { public function isLocal() { return true; } + + /** + * Acquire a lock on a file + */ + public function getLock($path, $lockType) { + return true; + } + + /** + * Release an existing lock + */ + public function releaseLock($path) { + return true; + } } } diff --git a/lib/private/files/storage/wrapper/lockingwrapper.php b/lib/private/files/storage/wrapper/lockingwrapper.php new file mode 100644 index 00000000000..04a325940bf --- /dev/null +++ b/lib/private/files/storage/wrapper/lockingwrapper.php @@ -0,0 +1,139 @@ +locks[$path])) { + $this->locks[$path] = new Lock($path); + } + $this->locks[$path]->addLock($lockType); + return $this->locks[$path]; + } + + /** + * Release an existing lock + * @param string $path Path to file, relative to this storage + * @return bool true on success, false on failure + */ + protected function releaseLock($path, $lockType, $releaseAll = false){ + $path = Filesystem::normalizePath($path); + if(isset($this->locks[$path])) { + if($releaseAll) { + return $this->locks[$path]->releaseAll(); + } + else { + return $this->locks[$path]->release($lockType); + } + } + return true; + } + + /** + * see http://php.net/manual/en/function.file_get_contents.php + * @param string $path + * @return string + * @throws \Exception + */ + public function file_get_contents($path) { + try { + if (!$this->getLock($path, Lock::READ)) { + throw new LockNotAcquiredException($path, Lock::READ); + } + $result = $this->storage->file_get_contents($path); + } + catch(\Exception $originalException) { + // Need to release the lock before more operations happen in upstream exception handlers + $this->releaseLock($path, Lock::READ); + throw $originalException; + } + return $result; + } + + public function file_put_contents($path, $data) { + try { + if (!$this->getLock($path, Lock::WRITE)) { + throw new LockNotAcquiredException($path, Lock::WRITE); + } + $result = $this->storage->file_put_contents($path, $data); + } + catch(\Exception $originalException) { + // Release lock, throw original exception + $this->releaseLock($path, Lock::WRITE); + throw $originalException; + } + return $result; + } + + + public function copy($path1, $path2) { + try { + if (!$this->getLock($path1, Lock::READ)) { + throw new LockNotAcquiredException($path1, Lock::READ); + } + if (!$this->getLock($path2, Lock::WRITE)) { + throw new LockNotAcquiredException($path2, Lock::WRITE); + } + $result = $this->storage->copy($path1, $path2); + } + catch(\Exception $originalException) { + // Release locks, throw original exception + $this->releaseLock($path1, Lock::READ); + $this->releaseLock($path2, Lock::WRITE); + throw $originalException; + } + return $result; + } + + public function rename($path1, $path2) { + try { + if (!$this->getLock($path1, Lock::READ)) { + throw new LockNotAcquiredException($path1, Lock::READ); + } + if (!$this->getLock($path2, Lock::WRITE)) { + throw new LockNotAcquiredException($path2, Lock::WRITE); + } + $result = $this->storage->rename($path1, $path2); + } + catch(\Exception $originalException) { + // Release locks, throw original exception + $this->releaseLock($path1, Lock::READ); + $this->releaseLock($path2, Lock::WRITE); + throw $originalException; + } + return $result; + } + + +} \ No newline at end of file diff --git a/lib/private/util.php b/lib/private/util.php index 306e37b9478..8cc48c03462 100755 --- a/lib/private/util.php +++ b/lib/private/util.php @@ -71,6 +71,16 @@ class OC_Util { return $storage; }); + // Set up flock + \OC\Files\Filesystem::addStorageWrapper(function($mountPoint, /** @var \OC\Files\Storage\Storage|null $storage */ $storage){ + // lock files on all local storage + if ($storage instanceof \OC\Files\Storage\Storage && $storage->isLocal()) { + return new \OC\Files\Storage\Wrapper\LockingWrapper(array('storage' => $storage)); + } else { + return $storage; + } + }); + $userDir = '/'.$user.'/files'; $userRoot = OC_User::getHome($user); $userDirectory = $userRoot . '/files'; -- cgit v1.2.3