2012-08-16 18:18:18 +00:00
< ? php
/**
* ownCloud
*
* @ author Robin Appelman
* @ copyright 2011 Robin Appelman icewind1991 @ gmail . com
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
* License as published by the Free Software Foundation ; either
* version 3 of the License , or any later version .
*
* This library 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 along with this library . If not , see < http :// www . gnu . org / licenses />.
*
*/
/**
* transparently encrypted filestream
*
* you can use it as wrapper around an existing stream by setting CryptStream :: $sourceStreams [ 'foo' ] = array ( 'path' => $path , 'stream' => $stream )
* and then fopen ( 'crypt://streams/foo' );
*/
namespace OCA_Encryption ;
class Stream {
public static $sourceStreams = array ();
private $source ;
private $path ;
2012-08-23 15:43:10 +00:00
private $rawPath ; // The raw path received by stream_open
2012-08-16 18:18:18 +00:00
private $readBuffer ; // For streams that dont support seeking
private $meta = array (); // Header / meta for source stream
private $count ;
private $writeCache ;
private $size ;
2012-08-23 15:43:10 +00:00
private $keyfile ;
2012-08-16 18:18:18 +00:00
private static $view ;
public function stream_open ( $path , $mode , $options , & $opened_path ) {
// Get access to filesystem via filesystemview object
if ( ! self :: $view ) {
self :: $view = new \OC_FilesystemView ( '' );
}
// Get the bare file path
$path = str_replace ( 'crypt://' , '' , $path );
2012-08-23 15:43:10 +00:00
$this -> rawPath = $path ;
2012-08-16 18:18:18 +00:00
if (
dirname ( $path ) == 'streams'
and isset ( self :: $sourceStreams [ basename ( $path )] )
) {
$this -> source = self :: $sourceStreams [ basename ( $path )][ 'stream' ];
$this -> path = self :: $sourceStreams [ basename ( $path )][ 'path' ];
$this -> size = self :: $sourceStreams [ basename ( $path )][ 'size' ];
} else {
if (
$mode == 'w'
or $mode == 'w+'
or $mode == 'wb'
or $mode == 'wb+'
) {
$this -> size = 0 ;
} else {
$this -> size = self :: $view -> filesize ( $path , $mode );
}
// Disable fileproxies so we can open the source file without recursive encryption
\OC_FileProxy :: $enabled = false ;
$this -> source = self :: $view -> fopen ( $path , $mode );
\OC_FileProxy :: $enabled = true ;
if ( ! is_resource ( $this -> source ) ) {
\OCP\Util :: writeLog ( 'files_encryption' , 'failed to open ' . $path , OCP\Util :: ERROR );
}
}
if ( is_resource ( $this -> source ) ) {
$this -> meta = stream_get_meta_data ( $this -> source );
}
return is_resource ( $this -> source );
}
public function stream_seek ( $offset , $whence = SEEK_SET ) {
$this -> flush ();
fseek ( $this -> source , $offset , $whence );
}
public function stream_tell () {
return ftell ( $this -> source );
}
2012-08-23 15:43:10 +00:00
public function stream_read ( $count ) {
$this -> writeCache = '' ;
if ( $count != 8192 ) {
// $count will always be 8192 https://bugs.php.net/bug.php?id=21641
// This makes this function a lot simpler, but will break this class if the above 'bug' gets 'fixed'
\OCP\Util :: writeLog ( 'files_encryption' , 'PHP "bug" 21641 no longer holds, decryption system requires refactoring' , OCP\Util :: FATAL );
2012-08-16 18:18:18 +00:00
die ();
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
}
2012-08-23 15:43:10 +00:00
$pos = ftell ( $this -> source );
$data = fread ( $this -> source , 8192 );
if ( strlen ( $data ) ) {
$result = Crypt :: symmetricDecryptFileContent ( $data , $this -> keyfile );
} else {
$result = '' ;
2012-08-16 18:18:18 +00:00
}
2012-08-23 15:43:10 +00:00
$length = $this -> size - $pos ;
if ( $length < 8192 ) {
$result = substr ( $result , 0 , $length );
2012-08-16 18:18:18 +00:00
}
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
return $result ;
2012-08-23 15:43:10 +00:00
}
/**
* @ brief Get the keyfile for the current file , generate one if necessary
*/
public function getKey () {
# TODO: Move this user call out of here - it belongs elsewhere
$user = \OCP\User :: getUser ();
if ( self :: $view -> file_exists ( $this -> rawPath . $user ) ) {
// If the data is to be written to an existing file, fetch its keyfile
$this -> keyfile = Keymanager :: getFileKey ( $this -> rawPath . $user );
} else {
// If the data is to be written to a new file, generate a new keyfile
$this -> keyfile = Crypt :: generateKey ();
}
2012-08-16 18:18:18 +00:00
}
/**
2012-08-23 18:19:39 +00:00
* @ brief Write write plan data as encrypted data
2012-08-16 18:18:18 +00:00
*/
public function stream_write ( $data ) {
2012-08-23 15:43:10 +00:00
# TODO: Find a way to get path of file in order to know where to save its parallel keyfile
\OC_FileProxy :: $enabled = false ;
2012-08-16 18:18:18 +00:00
$length = strlen ( $data );
$written = 0 ;
$currentPos = ftell ( $this -> source );
# TODO: Move this user call out of here - it belongs elsewhere
$user = \OCP\User :: getUser ();
2012-08-23 15:43:10 +00:00
// Set keyfile property for file in question
$this -> getKey ();
2012-08-16 18:18:18 +00:00
2012-08-23 15:43:10 +00:00
if ( ! self :: $view -> file_exists ( $this -> rawPath . $user ) ) {
2012-08-16 18:18:18 +00:00
2012-08-23 15:43:10 +00:00
// Save keyfile in parallel directory structure
Keymanager :: setFileKey ( $this -> rawPath , $this -> keyfile , new \OC_FilesystemView ( '/' ) );
2012-08-16 18:18:18 +00:00
}
2012-08-23 18:19:39 +00:00
// // Set $data to contents of writeCache
// // Concat writeCache to start of $data
// if ( $this->writeCache ) {
//
// $data = $this->writeCache . $data;
//
// $this->writeCache = '';
//
// }
// // Make sure we always start on a block start
// if ( 0 != ( $currentPos % 8192 ) ) { // If we're not at the end of file yet (in the final chunk), if there will be no bytes left to read after the current chunk
//
// fseek( $this->source, - ( $currentPos % 8192 ), SEEK_CUR );
//
// $encryptedBlock = fread( $this->source, 8192 );
//
// fseek( $this->source, - ( $currentPos % 8192 ), SEEK_CUR );
//
// $block = Crypt::symmetricDecryptFileContent( $encryptedBlock, $this->keyfile );
//
// $x = substr( $block, 0, $currentPos % 8192 );
//
// $data = $x . $data;
//
// fseek( $this->source, - ( $currentPos % 8192 ), SEEK_CUR );
//
// }
// $currentPos = ftell( $this->source );
//
// while( $remainingLength = strlen( $data ) > 0 ) {
//
// // Set writeCache to contents of $data
// if ( $remainingLength < 8192 ) {
//
// $this->writeCache = $data;
//
// $data = '';
//
// } else {
2012-08-23 15:43:10 +00:00
$encrypted = Crypt :: symmetricBlockEncryptFileContent ( $data , $this -> keyfile );
//$encrypted = $data;
2012-08-16 18:18:18 +00:00
2012-08-23 15:43:10 +00:00
fwrite ( $this -> source , $encrypted );
2012-08-16 18:18:18 +00:00
2012-08-23 18:19:39 +00:00
$data = substr ( $data , 8192 );
2012-08-16 18:18:18 +00:00
2012-08-23 18:19:39 +00:00
// }
//
// }
2012-08-16 18:18:18 +00:00
$this -> size = max ( $this -> size , $currentPos + $length );
return $length ;
}
public function stream_set_option ( $option , $arg1 , $arg2 ) {
switch ( $option ) {
case STREAM_OPTION_BLOCKING :
stream_set_blocking ( $this -> source , $arg1 );
break ;
case STREAM_OPTION_READ_TIMEOUT :
stream_set_timeout ( $this -> source , $arg1 , $arg2 );
break ;
case STREAM_OPTION_WRITE_BUFFER :
stream_set_write_buffer ( $this -> source , $arg1 , $arg2 );
}
}
public function stream_stat () {
return fstat ( $this -> source );
}
public function stream_lock ( $mode ) {
flock ( $this -> source , $mode );
}
public function stream_flush () {
return fflush ( $this -> source );
}
public function stream_eof () {
return feof ( $this -> source );
}
private function flush () {
if ( $this -> writeCache ) {
$encrypted = Crypt :: encrypt ( $this -> writeCache );
fwrite ( $this -> source , $encrypted );
$this -> writeCache = '' ;
}
}
public function stream_close () {
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
$this -> flush ();
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
if ( $this -> meta [ 'mode' ] != 'r' and $this -> meta [ 'mode' ] != 'rb' ) {
2012-08-23 15:43:10 +00:00
\OC_FileCache :: put ( $this -> path , array ( 'encrypted' => true , 'size' => $this -> size ), '' );
2012-08-16 18:18:18 +00:00
}
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
return fclose ( $this -> source );
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
}
2012-08-23 15:43:10 +00:00
2012-08-16 18:18:18 +00:00
}