test to simulate a non-seekable stream wrapper

This commit is contained in:
Bjoern Schiessle 2015-05-20 14:29:20 +02:00
parent fb51880a4a
commit 5a20edac82
3 changed files with 102 additions and 9 deletions

View file

@ -130,6 +130,7 @@ class Encryption extends Wrapper {
* @param int $size
* @param int $unencryptedSize
* @param int $headerSize
* @param string $wrapper stream wrapper class
* @return resource
*
* @throws \BadMethodCallException
@ -144,7 +145,8 @@ class Encryption extends Wrapper {
$mode,
$size,
$unencryptedSize,
$headerSize) {
$headerSize,
$wrapper = 'OC\Files\Stream\Encryption') {
$context = stream_context_create(array(
'ocencryption' => array(
@ -164,7 +166,7 @@ class Encryption extends Wrapper {
)
));
return self::wrapSource($source, $mode, $context, 'ocencryption', 'OC\Files\Stream\Encryption');
return self::wrapSource($source, $mode, $context, 'ocencryption', $wrapper);
}
/**
@ -309,7 +311,7 @@ class Encryption extends Wrapper {
// flush will start writing there when the position moves to another block
$positionInFile = (int)floor($this->position / $this->unencryptedBlockSize) *
$this->util->getBlockSize() + $this->headerSize;
$resultFseek = parent::stream_seek($positionInFile);
$resultFseek = $this->parentStreamSeek($positionInFile);
// only allow writes on seekable streams, or at the end of the encrypted stream
if (!($this->readOnly) && ($resultFseek || $positionInFile === $this->size)) {
@ -376,10 +378,10 @@ class Encryption extends Wrapper {
* $this->util->getBlockSize() + $this->headerSize;
$oldFilePosition = parent::stream_tell();
if (parent::stream_seek($newFilePosition)) {
parent::stream_seek($oldFilePosition);
if ($this->parentStreamSeek($newFilePosition)) {
$this->parentStreamSeek($oldFilePosition);
$this->flush();
parent::stream_seek($newFilePosition);
$this->parentStreamSeek($newFilePosition);
$this->position = $newPosition;
$return = true;
}
@ -456,4 +458,14 @@ class Encryption extends Wrapper {
parent::stream_read($this->headerSize);
}
/**
* call stream_seek() from parent class
*
* @param integer $position
* @return bool
*/
protected function parentStreamSeek($position) {
return parent::stream_seek($position);
}
}

View file

@ -0,0 +1,37 @@
<?php
/**
* @author Björn Schießle <schiessle@owncloud.com>
*
* @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 Test\Files\Stream;
class DummyEncryptionWrapper extends \OC\Files\Stream\Encryption {
/**
* simulate a non-seekable stream wrapper by always return false
*
* @param int $position
* @return bool
*/
protected function parentStreamSeek($position) {
return false;
}
}

View file

@ -16,7 +16,7 @@ class Encryption extends \Test\TestCase {
* @param integer $unencryptedSize
* @return resource
*/
protected function getStream($fileName, $mode, $unencryptedSize) {
protected function getStream($fileName, $mode, $unencryptedSize, $wrapper = '\OC\Files\Stream\Encryption') {
clearstatcache();
$size = filesize($fileName);
$source = fopen($fileName, $mode);
@ -45,9 +45,10 @@ class Encryption extends \Test\TestCase {
->method('getUidAndFilename')
->willReturn(['user1', $internalPath]);
return \OC\Files\Stream\Encryption::wrap($source, $internalPath,
return $wrapper::wrap($source, $internalPath,
$fullPath, $header, $uid, $this->encryptionModule, $storage, $encStorage,
$util, $file, $mode, $size, $unencryptedSize, 8192);
$util, $file, $mode, $size, $unencryptedSize, 8192, $wrapper);
}
/**
@ -256,6 +257,49 @@ class Encryption extends \Test\TestCase {
unlink($fileName);
}
/**
* simulate a non-seekable storage
*
* @dataProvider dataFilesProvider
*/
public function testWriteToNonSeekableStorage($testFile) {
$wrapper = $this->getMockBuilder('\OC\Files\Stream\Encryption')
->setMethods(['parentSeekStream'])->getMock();
$wrapper->expects($this->any())->method('parentSeekStream')->willReturn(false);
$expectedData = file_get_contents(\OC::$SERVERROOT . '/tests/data/' . $testFile);
// write it
$fileName = tempnam("/tmp", "FOO");
$stream = $this->getStream($fileName, 'w+', 0, '\Test\Files\Stream\DummyEncryptionWrapper');
// while writing the file from the beginning to the end we should never try
// to read parts of the file. This should only happen for write operations
// in the middle of a file
$this->encryptionModule->expects($this->never())->method('decrypt');
fwrite($stream, $expectedData);
fclose($stream);
// read it all
$stream = $this->getStream($fileName, 'r', strlen($expectedData), '\Test\Files\Stream\DummyEncryptionWrapper');
$data = stream_get_contents($stream);
fclose($stream);
$this->assertEquals($expectedData, $data);
// another read test with a loop like we do in several places:
$stream = $this->getStream($fileName, 'r', strlen($expectedData));
$data = '';
while (!feof($stream)) {
$data .= fread($stream, 8192);
}
fclose($stream);
$this->assertEquals($expectedData, $data);
unlink($fileName);
}
/**
* @return \PHPUnit_Framework_MockObject_MockObject
*/