* @author Thomas Müller * * @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 * */ namespace OC\Encryption\Keys; use OC\Encryption\Util; use OC\Files\View; use OCP\Encryption\Exceptions\GenericEncryptionException; class Storage implements \OCP\Encryption\Keys\IStorage { /** @var View */ private $view; /** @var Util */ private $util; // base dir where all the file related keys are stored private $keys_base_dir; private $encryption_base_dir; private $keyCache = array(); /** @var string */ private $encryptionModuleId; /** * @param string $encryptionModuleId * @param View $view * @param Util $util */ public function __construct($encryptionModuleId, View $view, Util $util) { $this->view = $view; $this->util = $util; $this->encryptionModuleId = $encryptionModuleId; $this->encryption_base_dir = '/files_encryption'; $this->keys_base_dir = $this->encryption_base_dir .'/keys'; } /** * get user specific key * * @param string $uid ID if the user for whom we want the key * @param string $keyId id of the key * * @return mixed key */ public function getUserKey($uid, $keyId) { $path = $this->constructUserKeyPath($keyId, $uid); return $this->getKey($path); } /** * get file specific key * * @param string $path path to file * @param string $keyId id of the key * * @return mixed key */ public function getFileKey($path, $keyId) { $keyDir = $this->getFileKeyDir($path); return $this->getKey($keyDir . $keyId); } /** * get system-wide encryption keys not related to a specific user, * e.g something like a key for public link shares * * @param string $keyId id of the key * * @return mixed key */ public function getSystemUserKey($keyId) { $path = $this->constructUserKeyPath($keyId); return $this->getKey($path); } /** * set user specific key * * @param string $uid ID if the user for whom we want the key * @param string $keyId id of the key * @param mixed $key */ public function setUserKey($uid, $keyId, $key) { $path = $this->constructUserKeyPath($keyId, $uid); return $this->setKey($path, $key); } /** * set file specific key * * @param string $path path to file * @param string $keyId id of the key * @param boolean */ public function setFileKey($path, $keyId, $key) { $keyDir = $this->getFileKeyDir($path); return $this->setKey($keyDir . $keyId, $key); } /** * set system-wide encryption keys not related to a specific user, * e.g something like a key for public link shares * * @param string $keyId id of the key * @param mixed $key * * @return mixed key */ public function setSystemUserKey($keyId, $key) { $path = $this->constructUserKeyPath($keyId); return $this->setKey($path, $key); } /** * delete user specific key * * @param string $uid ID if the user for whom we want to delete the key * @param string $keyId id of the key * * @return boolean False when the key could not be deleted */ public function deleteUserKey($uid, $keyId) { $path = $this->constructUserKeyPath($keyId, $uid); return !$this->view->file_exists($path) || $this->view->unlink($path); } /** * delete file specific key * * @param string $path path to file * @param string $keyId id of the key * * @return boolean False when the key could not be deleted */ public function deleteFileKey($path, $keyId) { $keyDir = $this->getFileKeyDir($path); return !$this->view->file_exists($keyDir . $keyId) || $this->view->unlink($keyDir . $keyId); } /** * delete all file keys for a given file * * @param string $path to the file * @return boolean False when the key could not be deleted */ public function deleteAllFileKeys($path) { $keyDir = $this->getFileKeyDir($path); $path = dirname($keyDir); return !$this->view->file_exists($path) || $this->view->deleteAll($path); } /** * delete system-wide encryption keys not related to a specific user, * e.g something like a key for public link shares * * @param string $keyId id of the key * * @return boolean False when the key could not be deleted */ public function deleteSystemUserKey($keyId) { $path = $this->constructUserKeyPath($keyId); return !$this->view->file_exists($path) || $this->view->unlink($path); } /** * construct path to users key * * @param string $keyId * @param string $uid * @return string */ protected function constructUserKeyPath($keyId, $uid = null) { if ($uid === null) { $path = $this->encryption_base_dir . '/' . $this->encryptionModuleId . '/' . $keyId; } else { $path = '/' . $uid . $this->encryption_base_dir . '/' . $this->encryptionModuleId . '/' . $uid . '.' . $keyId; } return $path; } /** * read key from hard disk * * @param string $path to key * @return string */ private function getKey($path) { $key = ''; if ($this->view->file_exists($path)) { if (isset($this->keyCache[$path])) { $key = $this->keyCache[$path]; } else { $key = $this->view->file_get_contents($path); $this->keyCache[$path] = $key; } } return $key; } /** * write key to disk * * * @param string $path path to key directory * @param string $key key * @return bool */ private function setKey($path, $key) { $this->keySetPreparation(dirname($path)); $result = $this->view->file_put_contents($path, $key); if (is_int($result) && $result > 0) { $this->keyCache[$path] = $key; return true; } return false; } /** * get path to key folder for a given file * * @param string $path path to the file, relative to data/ * @return string * @throws GenericEncryptionException * @internal param string $keyId */ private function getFileKeyDir($path) { if ($this->view->is_dir($path)) { throw new GenericEncryptionException("file was expected but directory was given: $path"); } list($owner, $filename) = $this->util->getUidAndFilename($path); $filename = $this->util->stripPartialFileExtension($filename); // in case of system wide mount points the keys are stored directly in the data directory if ($this->util->isSystemWideMountPoint($filename)) { $keyPath = $this->keys_base_dir . $filename . '/'; } else { $keyPath = '/' . $owner . $this->keys_base_dir . $filename . '/'; } return \OC\Files\Filesystem::normalizePath($keyPath . $this->encryptionModuleId . '/', false); } /** * move keys if a file was renamed * * @param string $source * @param string $target * @param string $owner * @param bool $systemWide */ public function renameKeys($source, $target) { list($owner, $source) = $this->util->getUidAndFilename($source); list(, $target) = $this->util->getUidAndFilename($target); $systemWide = $this->util->isSystemWideMountPoint($target); if ($systemWide) { $sourcePath = $this->keys_base_dir . $source . '/'; $targetPath = $this->keys_base_dir . $target . '/'; } else { $sourcePath = '/' . $owner . $this->keys_base_dir . $source . '/'; $targetPath = '/' . $owner . $this->keys_base_dir . $target . '/'; } if ($this->view->file_exists($sourcePath)) { $this->keySetPreparation(dirname($targetPath)); $this->view->rename($sourcePath, $targetPath); } } /** * copy keys if a file was renamed * * @param string $source * @param string $target * @param string $owner * @param bool $systemWide */ public function copyKeys($source, $target) { list($owner, $source) = $this->util->getUidAndFilename($source); list(, $target) = $this->util->getUidAndFilename($target); $systemWide = $this->util->isSystemWideMountPoint($target); if ($systemWide) { $sourcePath = $this->keys_base_dir . $source . '/'; $targetPath = $this->keys_base_dir . $target . '/'; } else { $sourcePath = '/' . $owner . $this->keys_base_dir . $source . '/'; $targetPath = '/' . $owner . $this->keys_base_dir . $target . '/'; } if ($this->view->file_exists($sourcePath)) { $this->keySetPreparation(dirname($targetPath)); $this->view->copy($sourcePath, $targetPath); } } /** * Make preparations to filesystem for saving a keyfile * * @param string $path relative to the views root */ protected function keySetPreparation($path) { // If the file resides within a subdirectory, create it if (!$this->view->file_exists($path)) { $sub_dirs = explode('/', ltrim($path, '/')); $dir = ''; foreach ($sub_dirs as $sub_dir) { $dir .= '/' . $sub_dir; if (!$this->view->is_dir($dir)) { $this->view->mkdir($dir); } } } } }