395 lines
9.3 KiB
PHP
395 lines
9.3 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* @author Clark Tomlinson <clark@owncloud.com>
|
||
|
* @since 3/17/15, 10:31 AM
|
||
|
* @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\Encryption;
|
||
|
|
||
|
|
||
|
use OC\Files\Filesystem;
|
||
|
use OC\Files\View;
|
||
|
use OCA\Encryption\Crypto\Crypt;
|
||
|
use OCA\Files_Versions\Storage;
|
||
|
use OCP\App;
|
||
|
use OCP\IConfig;
|
||
|
use OCP\ILogger;
|
||
|
use OCP\IUser;
|
||
|
use OCP\IUserSession;
|
||
|
use OCP\PreConditionNotMetException;
|
||
|
use OCP\Share;
|
||
|
|
||
|
class Util {
|
||
|
/**
|
||
|
* @var View
|
||
|
*/
|
||
|
private $files;
|
||
|
/**
|
||
|
* @var Filesystem
|
||
|
*/
|
||
|
private $filesystem;
|
||
|
/**
|
||
|
* @var Crypt
|
||
|
*/
|
||
|
private $crypt;
|
||
|
/**
|
||
|
* @var KeyManager
|
||
|
*/
|
||
|
private $keyManager;
|
||
|
/**
|
||
|
* @var ILogger
|
||
|
*/
|
||
|
private $logger;
|
||
|
/**
|
||
|
* @var bool|IUser
|
||
|
*/
|
||
|
private $user;
|
||
|
/**
|
||
|
* @var IConfig
|
||
|
*/
|
||
|
private $config;
|
||
|
|
||
|
/**
|
||
|
* Util constructor.
|
||
|
*
|
||
|
* @param View $files
|
||
|
* @param Filesystem $filesystem
|
||
|
* @param Crypt $crypt
|
||
|
* @param KeyManager $keyManager
|
||
|
* @param ILogger $logger
|
||
|
* @param IUserSession $userSession
|
||
|
* @param IConfig $config
|
||
|
*/
|
||
|
public function __construct(
|
||
|
View $files,
|
||
|
Filesystem $filesystem,
|
||
|
Crypt $crypt,
|
||
|
KeyManager $keyManager,
|
||
|
ILogger $logger,
|
||
|
IUserSession $userSession,
|
||
|
IConfig $config
|
||
|
) {
|
||
|
$this->files = $files;
|
||
|
$this->filesystem = $filesystem;
|
||
|
$this->crypt = $crypt;
|
||
|
$this->keyManager = $keyManager;
|
||
|
$this->logger = $logger;
|
||
|
$this->user = $userSession && $userSession->isLoggedIn() ? $userSession->getUser() : false;
|
||
|
$this->config = $config;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $dirPath
|
||
|
* @param bool $found
|
||
|
* @return array|bool
|
||
|
*/
|
||
|
private function findEncryptedFiles($dirPath, &$found = false) {
|
||
|
|
||
|
if ($found === false) {
|
||
|
$found = [
|
||
|
'plain' => [],
|
||
|
'encrypted' => [],
|
||
|
'broken' => [],
|
||
|
];
|
||
|
}
|
||
|
|
||
|
if ($this->files->is_dir($dirPath) && $handle = $this->files->opendir($dirPath)) {
|
||
|
if (is_resource($handle)) {
|
||
|
while (($file = readdir($handle) !== false)) {
|
||
|
if ($file !== '.' && $file !== '..') {
|
||
|
|
||
|
// Skip stray part files
|
||
|
if ($this->isPartialFilePath($file)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$filePath = $dirPath . '/' . $this->files->getRelativePath('/' . $file);
|
||
|
$relPath = $this->stripUserFilesPath($filePath);
|
||
|
|
||
|
// If the path is a directory, search its contents
|
||
|
if ($this->files->is_dir($filePath)) {
|
||
|
// Recurse back
|
||
|
$this->findEncryptedFiles($filePath);
|
||
|
|
||
|
/*
|
||
|
* If the path is a file,
|
||
|
* determine where they got re-enabled :/
|
||
|
*/
|
||
|
} elseif ($this->files->is_file($filePath)) {
|
||
|
$isEncryptedPath = $this->isEncryptedPath($filePath);
|
||
|
|
||
|
/**
|
||
|
* If the file is encrypted
|
||
|
*
|
||
|
* @note: if the userId is
|
||
|
* empty or not set, file will
|
||
|
* be detected as plain
|
||
|
* @note: this is inefficient;
|
||
|
* scanning every file like this
|
||
|
* will eat server resources :(
|
||
|
* fixMe: xxx find better way
|
||
|
*/
|
||
|
if ($isEncryptedPath) {
|
||
|
$fileKey = $this->keyManager->getFileKey($relPath);
|
||
|
$shareKey = $this->keyManager->getShareKey($relPath);
|
||
|
// If file is encrypted but now file key is available, throw exception
|
||
|
if (!$fileKey || !$shareKey) {
|
||
|
$this->logger->error('Encryption library, no keys avilable to decrypt the file: ' . $file);
|
||
|
$found['broken'][] = [
|
||
|
'name' => $file,
|
||
|
'path' => $filePath,
|
||
|
];
|
||
|
} else {
|
||
|
$found['encrypted'][] = [
|
||
|
'name' => $file,
|
||
|
'path' => $filePath
|
||
|
];
|
||
|
}
|
||
|
} else {
|
||
|
$found['plain'][] = [
|
||
|
'name' => $file,
|
||
|
'path' => $filePath
|
||
|
];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $found;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $path
|
||
|
* @return bool
|
||
|
*/
|
||
|
private function isPartialFilePath($path) {
|
||
|
$extension = pathinfo($path, PATHINFO_EXTENSION);
|
||
|
|
||
|
if ($extension === 'part') {
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $filePath
|
||
|
* @return bool|string
|
||
|
*/
|
||
|
private function stripUserFilesPath($filePath) {
|
||
|
$split = $this->splitPath($filePath);
|
||
|
|
||
|
// It is not a file relative to data/user/files
|
||
|
if (count($split) < 4 || $split[2] !== 'files') {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$sliced = array_slice($split, 3);
|
||
|
|
||
|
return implode('/', $sliced);
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $filePath
|
||
|
* @return array
|
||
|
*/
|
||
|
private function splitPath($filePath) {
|
||
|
$normalized = $this->filesystem->normalizePath($filePath);
|
||
|
|
||
|
return explode('/', $normalized);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $filePath
|
||
|
* @return bool
|
||
|
*/
|
||
|
private function isEncryptedPath($filePath) {
|
||
|
$data = '';
|
||
|
|
||
|
// We only need 24 bytes from the last chunck
|
||
|
if ($this->files->file_exists($filePath)) {
|
||
|
$handle = $this->files->fopen($filePath, 'r');
|
||
|
if (is_resource($handle)) {
|
||
|
// Suppress fseek warning, we handle the case that fseek
|
||
|
// doesn't work in the else branch
|
||
|
if (@fseek($handle, -24, SEEK_END) === 0) {
|
||
|
$data = fgets($handle);
|
||
|
} else {
|
||
|
// if fseek failed on the storage we create a local copy
|
||
|
// from the file and read this one
|
||
|
fclose($handle);
|
||
|
$localFile = $this->files->getLocalFile($filePath);
|
||
|
$handle = fopen($localFile, 'r');
|
||
|
|
||
|
if (is_resource($handle) && fseek($handle,
|
||
|
-24,
|
||
|
SEEK_END) === 0
|
||
|
) {
|
||
|
$data = fgets($handle);
|
||
|
}
|
||
|
}
|
||
|
fclose($handle);
|
||
|
return $this->crypt->isCatfileContent($data);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function recoveryEnabledForUser() {
|
||
|
$recoveryMode = $this->config->getUserValue($this->user->getUID(),
|
||
|
'encryption',
|
||
|
'recoveryEnabled',
|
||
|
0);
|
||
|
|
||
|
return ($recoveryMode === '1');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $enabled
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function setRecoveryForUser($enabled) {
|
||
|
$value = $enabled ? '1' : '0';
|
||
|
|
||
|
try {
|
||
|
$this->config->setUserValue($this->user->getUID(),
|
||
|
'encryption',
|
||
|
'recoveryEnabled',
|
||
|
$value);
|
||
|
return true;
|
||
|
} catch (PreConditionNotMetException $e) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $recoveryPassword
|
||
|
*/
|
||
|
public function recoverUsersFiles($recoveryPassword) {
|
||
|
// todo: get system private key here
|
||
|
// $this->keyManager->get
|
||
|
$privateKey = $this->crypt->decryptPrivateKey($encryptedKey,
|
||
|
$recoveryPassword);
|
||
|
|
||
|
$this->recoverAllFiles('/', $privateKey);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $uid
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function userHasFiles($uid) {
|
||
|
return $this->files->file_exists($uid . '/files');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $path
|
||
|
* @param $privateKey
|
||
|
*/
|
||
|
private function recoverAllFiles($path, $privateKey) {
|
||
|
// Todo relocate to storage
|
||
|
$dirContent = $this->files->getDirectoryContent($path);
|
||
|
|
||
|
foreach ($dirContent as $item) {
|
||
|
// Get relative path from encryption/keyfiles
|
||
|
$filePath = substr($item['path'], strlen('encryption/keys'));
|
||
|
if ($this->files->is_dir($this->user->getUID() . '/files' . '/' . $filePath)) {
|
||
|
$this->recoverAllFiles($filePath . '/', $privateKey);
|
||
|
} else {
|
||
|
$this->recoverFile($filePath, $privateKey);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $filePath
|
||
|
* @param $privateKey
|
||
|
*/
|
||
|
private function recoverFile($filePath, $privateKey) {
|
||
|
$sharingEnabled = Share::isEnabled();
|
||
|
|
||
|
// Find out who, if anyone, is sharing the file
|
||
|
if ($sharingEnabled) {
|
||
|
$result = Share::getUsersSharingFile($filePath,
|
||
|
$this->user->getUID(),
|
||
|
true);
|
||
|
$userIds = $result['users'];
|
||
|
$userIds[] = 'public';
|
||
|
} else {
|
||
|
$userIds = [
|
||
|
$this->user->getUID(),
|
||
|
$this->recoveryKeyId
|
||
|
];
|
||
|
}
|
||
|
$filteredUids = $this->filterShareReadyUsers($userIds);
|
||
|
|
||
|
// Decrypt file key
|
||
|
$encKeyFile = $this->keyManager->getFileKey($filePath);
|
||
|
$shareKey = $this->keyManager->getShareKey($filePath);
|
||
|
$plainKeyFile = $this->crypt->multiKeyDecrypt($encKeyFile,
|
||
|
$shareKey,
|
||
|
$privateKey);
|
||
|
|
||
|
// Encrypt the file key again to all users, this time with the new publick keyt for the recovered user
|
||
|
$userPublicKeys = $this->keyManager->getPublicKeys($filteredUids['ready']);
|
||
|
$multiEncryptionKey = $this->crypt->multiKeyEncrypt($plainKeyFile,
|
||
|
$userPublicKeys);
|
||
|
|
||
|
$this->keyManager->setFileKeys($multiEncryptionKey['data']);
|
||
|
$this->keyManager->setShareKeys($multiEncryptionKey['keys']);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $userIds
|
||
|
* @return array
|
||
|
*/
|
||
|
private function filterShareReadyUsers($userIds) {
|
||
|
// This array will collect the filtered IDs
|
||
|
$readyIds = $unreadyIds = [];
|
||
|
|
||
|
// Loop though users and create array of UIDs that need new keyfiles
|
||
|
foreach ($userIds as $user) {
|
||
|
// Check that the user is encryption capable, or is the
|
||
|
// public system user (for public shares)
|
||
|
if ($this->isUserReady($user)) {
|
||
|
// construct array of ready UIDs for keymanager
|
||
|
$readyIds[] = $user;
|
||
|
} else {
|
||
|
// Construct array of unready UIDs for keymanager
|
||
|
$unreadyIds[] = $user;
|
||
|
|
||
|
// Log warning; we cant do necessary setup here
|
||
|
// because we don't have the user passphrase
|
||
|
$this->logger->warning('Encryption Library ' . $this->user->getUID() . ' is not setup for encryption');
|
||
|
}
|
||
|
}
|
||
|
return [
|
||
|
'ready' => $readyIds,
|
||
|
'unready' => $unreadyIds
|
||
|
];
|
||
|
}
|
||
|
|
||
|
}
|