summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--apps/files_encryption/lib/util.php73
1 files changed, 45 insertions, 28 deletions
diff --git a/apps/files_encryption/lib/util.php b/apps/files_encryption/lib/util.php
index 33c2f88b0fd..3cf83703295 100644
--- a/apps/files_encryption/lib/util.php
+++ b/apps/files_encryption/lib/util.php
@@ -385,15 +385,24 @@ class Util {
&& $this->isEncryptedPath($path)
) {
- $offset = 0;
+ $cipher = Helper::getCipher();
+ $realSize = 0;
+
+ // get the size from filesystem
+ $size = $this->view->filesize($path);
+
+ // open stream
+ $stream = fopen($path, "r");
+
+ // if the file contains a encryption header we
+ // we set the cipher
+ // and we update the size
if ($this->containHeader($path)) {
- $offset = Crypt::BLOCKSIZE;
+ $header = fread($stream,Crypt::BLOCKSIZE);
+ $cipher = Crypt::getCipher($header);
+ $size -= Crypt::BLOCKSIZE;
}
- // get the size from filesystem if the file contains a encryption header we
- // we substract it
- $size = $this->view->filesize($path) - $offset;
-
// fast path, else the calculation for $lastChunkNr is bogus
if ($size === 0) {
\OC_FileProxy::$enabled = $proxyStatus;
@@ -403,37 +412,45 @@ class Util {
// calculate last chunk nr
// next highest is end of chunks, one subtracted is last one
// we have to read the last chunk, we can't just calculate it (because of padding etc)
- $lastChunkNr = ceil($size/ Crypt::BLOCKSIZE) - 1;
- $lastChunkSize = $size - ($lastChunkNr * Crypt::BLOCKSIZE);
-
- // open stream
- $stream = fopen('crypt://' . $path, "r");
+ $lastChunkNr = ceil($size/Crypt::BLOCKSIZE)-1;
if (is_resource($stream)) {
// calculate last chunk position
- $lastChunckPos = ($lastChunkNr * Crypt::BLOCKSIZE);
+ $lastChunkPos = ($lastChunkNr * Crypt::BLOCKSIZE);
- // seek to end
- if (@fseek($stream, $lastChunckPos) === -1) {
- // storage doesn't support fseek, we need a local copy
- fclose($stream);
- $localFile = $this->view->getLocalFile($path);
- Helper::addTmpFileToMapper($localFile, $path);
- $stream = fopen('crypt://' . $localFile, "r");
- if (fseek($stream, $lastChunckPos) === -1) {
- // if fseek also fails on the local storage, than
- // there is nothing we can do
- fclose($stream);
- \OCP\Util::writeLog('Encryption library', 'couldn\'t determine size of "' . $path, \OCP\Util::ERROR);
- return $result;
+ // get the content of the last chunk
+ $lastChunkContentEncrypted='';
+ $count=Crypt::BLOCKSIZE;
+ if (@fseek($stream, $lastChunkPos, SEEK_CUR) === 0) {
+ $realSize+=$lastChunkNr*6126;
+ while ($count>0) {
+ $data=fread($stream,Crypt::BLOCKSIZE);
+ $count=strlen($data);
+ $lastChunkContentEncrypted.=$data;
+ }
+ } else {
+ while ($count>0) {
+ if(strlen($lastChunkContentEncrypted)>Crypt::BLOCKSIZE) {
+ $realSize+=6126;
+ $lastChunkContentEncrypted=substr($lastChunkContentEncrypted,Crypt::BLOCKSIZE);
+ }
+ $data=fread($stream,Crypt::BLOCKSIZE);
+ $count=strlen($data);
+ $lastChunkContentEncrypted.=$data;
}
}
- // get the content of the last chunk
- $lastChunkContent = fread($stream, $lastChunkSize);
+ $session = new \OCA\Encryption\Session(new \OC\Files\View('/'));
+ $privateKey = $session->getPrivateKey();
+ $plainKeyfile = $this->decryptKeyfile($path, $privateKey);
+ $shareKey = Keymanager::getShareKey($this->view, $this->keyId, $this, $path);
+
+ $plainKey = Crypt::multiKeyDecrypt($plainKeyfile, $shareKey, $privateKey);
+
+ $lastChunkContent=Crypt::symmetricDecryptFileContent($lastChunkContentEncrypted, $plainKey, $cipher);
// calc the real file size with the size of the last chunk
- $realSize = (($lastChunkNr * 6126) + strlen($lastChunkContent));
+ $realSize += strlen($lastChunkContent);
// store file size
$result = $realSize;