. */ /** * Privde a common interface to all different storage options * * Hooks provided: * read(storage,path) * write(storage,path) * create(storage,path) (when a file is created both, write and create will be emited) * delete(storage,path) * rename(storage,oldpath,newpath) */ class OC_FILESTORAGE{ public function __construct($parameters){} public function mkdir($path){} public function rmdir($path){} public function opendir($path){} public function is_dir($path){} public function is_file($path){} public function stat($path){} public function filetype($path){} public function filesize($path){} public function is_readable($path){} public function is_writeable($path){} public function file_exists($path){} public function readfile($path){} public function filectime($path){} public function filemtime($path){} public function fileatime($path){} public function file_get_contents($path){} public function file_put_contents($path,$data){} public function unlink($path){} public function rename($path1,$path2){} public function copy($path1,$path2){} public function fopen($path,$mode){} public function toTmpFile($path){}//copy the file to a temporary file, used for cross-storage file actions public function fromTmpFile($tmpPath,$path){}//copy a file from a temporary file, used for cross-storage file actions public function fromUploadedFile($tmpPath,$path){}//copy a file from a temporary file, used for cross-storage file actions public function getMimeType($path){} public function delTree($path){} public function find($path){} public function getTree($path){} public function hash($type,$path,$raw){} public function free_space($path){} } OC_FILESYSTEM::registerStorageType('local','OC_FILESTORAGE_LOCAL',array('datadir'=>'string')); /** * for local filestore, we only have to map the paths */ class OC_FILESTORAGE_LOCAL extends OC_FILESTORAGE{ private $datadir; public function __construct($arguments){ $this->datadir=$arguments['datadir']; if(substr($this->datadir,-1)!=='/'){ $this->datadir.='/'; } } public function mkdir($path){ if($return=mkdir($this->datadir.$path)){ $this->clearFolderSizeCache($path); } return $return; } public function rmdir($path){ if($return=rmdir($this->datadir.$path)){ OC_HOOK::emit( 'OC_FILESTORAGE', "delete", array( 'storage'=>$this, 'path' => $path)); $this->clearFolderSizeCache($path); } return $return; } public function opendir($path){ if($return=opendir($this->datadir.$path)){ OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); } return $return; } public function is_dir($path){ return (is_dir($this->datadir.$path) or substr($path,-1)=='/'); } public function is_file($path){ return is_file($this->datadir.$path); } public function stat($path){ return stat($this->datadir.$path); } public function filetype($path){ $filetype=filetype($this->datadir.$path); if($filetype=='link'){ $filetype=filetype(readlink($this->datadir.$path)); } return $filetype; } public function filesize($path){ if($this->is_dir($path)){ return $this->getFolderSize($path); }else{ return filesize($this->datadir.$path); } } public function is_readable($path){ return is_readable($this->datadir.$path); } public function is_writeable($path){ return is_writeable($this->datadir.$path); } public function file_exists($path){ return file_exists($this->datadir.$path); } public function readfile($path){ if($return=readfile($this->datadir.$path)){ OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); } return $return; } public function filectime($path){ return filectime($this->datadir.$path); } public function filemtime($path){ return filemtime($this->datadir.$path); } public function fileatime($path){ return fileatime($this->datadir.$path); } public function file_get_contents($path){ if($return=file_get_contents($this->datadir.$path)){ OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); } return $return; } public function file_put_contents($path,$data){ if(!$this->file_exists($path)){ OC_HOOK::emit( 'OC_FILESTORAGE', 'create', array( 'storage'=>$this, 'path' => $path)); } if($return=file_put_contents($this->datadir.$path,$data)){ OC_HOOK::emit( 'OC_FILESTORAGE', "write", array( 'storage'=>$this, 'path' => $path)); $this->clearFolderSizeCache($path); } } public function unlink($path){ if($return=unlink($this->datadir.$path)){ OC_HOOK::emit( 'OC_FILESTORAGE', "delete", array( 'storage'=>$this, 'path' => $path)); $this->clearFolderSizeCache($path); } return $return; } public function rename($path1,$path2){ if($return=rename($this->datadir.$path1,$this->datadir.$path2)){ OC_HOOK::emit( 'OC_FILESTORAGE', "rename", array( 'storage'=>$this, 'oldpath' => $path1, 'newpath' => $path2)); } return $return; } public function copy($path1,$path2){ if($this->is_dir($path2)){ if(!$this->file_exists($path2)){ $this->mkdir($path2); } $source=substr($path1,strrpos($path1,'/')+1); $path2.=$source; } if($return=copy($this->datadir.$path1,$this->datadir.$path2)){ OC_HOOK::emit( 'OC_FILESTORAGE', "create", array( 'storage'=>$this, 'path' => $path2)); $this->clearFolderSizeCache($path); } return $return; } public function fopen($path,$mode){ $exists=$this->file_exists($path); if($return=fopen($this->datadir.$path,$mode)){ switch($mode){ case 'r': OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); break; case 'r+': case 'w+': case 'x+': case 'a+': OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); OC_HOOK::emit( 'OC_FILESTORAGE', "write", array( 'storage'=>$this, 'path' => $path)); if(!$exists){ OC_HOOK::emit( 'OC_FILESTORAGE', 'create', array( 'storage'=>$this, 'path' => $path)); } $this->clearFolderSizeCache($path); break; case 'w': case 'x': case 'a': OC_HOOK::emit( 'OC_FILESTORAGE', "write", array( 'storage'=>$this, 'path' => $path)); if(!$exists){ OC_HOOK::emit( 'OC_FILESTORAGE', 'create', array( 'storage'=>$this, 'path' => $path)); } $this->clearFolderSizeCache($path); break; } } return $return; } public function getMimeType($fspath){ if($this->is_readable($fspath)){ if (@is_dir($this->datadir.$fspath)) { // directories are easy return "httpd/unix-directory"; }elseif (function_exists('finfo_open') and function_exists('finfo_file') and $finfo=finfo_open(FILEINFO_MIME)){ $mimeType =strtolower(finfo_file($finfo,$this->datadir.$fspath)); $mimeType=substr($mimeType,0,strpos($mimeType,';')); finfo_close($finfo); return $mimeType; } else if (function_exists("mime_content_type")) { // use mime magic extension if available $mime_type = mime_content_type($this->datadir.$fspath); } else if (self::canExecute("file")) { // it looks like we have a 'file' command, // lets see it it does have mime support $fp = popen("file -i -b '{$this->datadir}$fspath' 2>/dev/null", "r"); $reply = fgets($fp); pclose($fp); //trim the character set from the end of the response $mime_type=substr($reply,0,strrpos($reply,' ')); } if (empty($mime_type)) { // Fallback solution: try to guess the type by the file extension // TODO: add more ... switch (strtolower(strrchr(basename($fspath), "."))) { case '.css': $mime_type = 'text/css'; break; case '.flac': $mime_type = 'audio/flac'; break; case '.gif': $mime_type = 'image/gif'; break; case '.gzip': case '.gz': $mime_type = 'application/x-gzip'; break; case '.htm': case '.html': $mime_type = 'text/html'; break; case '.jpeg': case '.jpg': $mime_type = 'image/jpeg'; break; case '.js': $mime_type = 'application/x-javascript'; break; case '.oga': case '.ogg': $mime_type = 'audio/ogg'; break; case '.ogv': $mime_type = 'video/ogg'; break; case '.pdf': $mime_type = 'application/pdf'; break; case '.png': $mime_type = 'image/png'; break; case '.svg': $mime_type = 'image/svg+xml'; break; case '.tar': $mime_type = 'application/x-tar'; break; case '.tgz': $mime_type = 'application/x-compressed'; break; case '.tif': case '.tiff': $mime_type = 'image/tiff'; break; case '.txt': $mime_type = 'text/plain'; break; case '.zip': $mime_type = 'application/zip'; break; default: $mime_type = 'application/octet-stream'; break; } } return $mime_type; } } /** * detect if a given program is found in the search PATH * * helper function used by _mimetype() to detect if the * external 'file' utility is available * * @param string program name * @param string optional search path, defaults to $PATH * @return bool true if executable program found in path */ private function canExecute($name, $path = false) { // path defaults to PATH from environment if not set if ($path === false) { $path = getenv("PATH"); } // check method depends on operating system if (!strncmp(PHP_OS, "WIN", 3)) { // on Windows an appropriate COM or EXE file needs to exist $exts = array(".exe", ".com"); $check_fn = "file_exists"; } else { // anywhere else we look for an executable file of that name $exts = array(""); $check_fn = "is_executable"; } // Default check will be done with $path directories : $dirs = explode(PATH_SEPARATOR, $path); // WARNING : We have to check if open_basedir is enabled : $obd = ini_get('open_basedir'); if($obd != "none") $obd_values = explode(PATH_SEPARATOR, $obd); if(count($obd_values) > 0) { // open_basedir is in effect ! // We need to check if the program is in one of these dirs : $dirs = $obd_values; } foreach($dirs as $dir) { foreach($exts as $ext) { if($check_fn("$dir/$name".$ext)) return true; } } return false; } public function toTmpFile($path){ $tmpFolder=sys_get_temp_dir(); $filename=tempnam($tmpFolder,'OC_TEMP_FILE_'.substr($path,strrpos($path,'.'))); $fileStats = stat($this->datadir.$path); if(copy($this->datadir.$path,$filename)){ touch($filename, $fileStats['mtime'], $fileStats['atime']); OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); return $filename; }else{ return false; } } public function fromTmpFile($tmpFile,$path){ $fileStats = stat($tmpFile); if(rename($tmpFile,$this->datadir.$path)){ touch($this->datadir.$path, $fileStats['mtime'], $fileStats['atime']); OC_HOOK::emit( 'OC_FILESTORAGE', "create", array( 'storage'=>$this, 'path' => $path)); $this->clearFolderSizeCache($path); return true; }else{ return false; } } public function fromUploadedFile($tmpFile,$path){ $fileStats = stat($tmpFile); if(move_uploaded_file($tmpFile,$this->datadir.$path)){ touch($this->datadir.$path, $fileStats['mtime'], $fileStats['atime']); OC_HOOK::emit( 'OC_FILESTORAGE', "create", array( 'storage'=>$this, 'path' => $path)); $this->clearFolderSizeCache($path); return true; }else{ return false; } } public function delTree($dir) { $dirRelative=$dir; $dir=$this->datadir.$dir; if (!file_exists($dir)) return true; if (!is_dir($dir) || is_link($dir)) return unlink($dir); foreach (scandir($dir) as $item) { if ($item == '.' || $item == '..') continue; if(is_file($dir.'/'.$item)){ if(unlink($dir.'/'.$item)){ OC_HOOK::emit( 'OC_FILESTORAGE', "delete", array( 'storage'=>$this, 'path' => $dir.'/'.$item)); $this->clearFolderSizeCache($path); } }elseif(is_dir($dir.'/'.$item)){ if (!$this->delTree($dirRelative. "/" . $item)){ return false; }; } } if($return=rmdir($dir)){ OC_HOOK::emit( 'OC_FILESTORAGE', "delete", array( 'storage'=>$this, 'path' => $dir)); $this->clearFolderSizeCache($path); } return $return; } public function find($path){ $return=System::find($this->datadir.$path); foreach($return as &$file){ $file=str_replace($file,$this->datadir,''); } return $return; } public function getTree($dir) { if(substr($dir,-1,1)=='/'){ $dir=substr($dir,0,-1); } $tree=array(); $tree[]=$dir; $dirRelative=$dir; $dir=$this->datadir.$dir; if (!file_exists($dir)) return true; foreach (scandir($dir) as $item) { if ($item == '.' || $item == '..') continue; if(is_file($dir.'/'.$item)){ $tree[]=$dirRelative.'/'.$item; }elseif(is_dir($dir.'/'.$item)){ if ($subTree=$this->getTree($dirRelative. "/" . $item)){ $tree=array_merge($tree,$subTree); } } } return $tree; } public function hash($type,$path,$raw){ if($return=hash_file($type,$this->datadir.$path,$raw)){ OC_HOOK::emit( 'OC_FILESTORAGE', "read", array( 'storage'=>$this, 'path' => $path)); } return $return; } public function free_space($path){ return disk_free_space($this->datadir.$path); } /** * @brief get the size of folder and it's content * @param string $path file path * @return int size of folder and it's content */ public function getFolderSize($path){ $path=str_replace('//','/',$path); if($this->is_dir($path) and substr($path,-1)!='/'){ $path.='/'; } $query=OC_DB::prepare("SELECT size FROM *PREFIX*foldersize WHERE path=?"); $size=$query->execute(array($path))->fetchAll(); if(count($size)>0){// we already the size, just return it return $size[0]['size']; }else{//the size of the folder isn't know, calulate it return $this->calculateFolderSize($path); } } /** * @brief calulate the size of folder and it's content and cache it * @param string $path file path * @return int size of folder and it's content */ public function calculateFolderSize($path){ if($this->is_file($path)){ $path=dirname($path); } $path=str_replace('//','/',$path); if($this->is_dir($path) and substr($path,-1)!='/'){ $path.='/'; } $size=0; if ($dh = $this->opendir($path)) { while (($filename = readdir($dh)) !== false) { if($filename!='.' and $filename!='..'){ $subFile=$path.'/'.$filename; if($this->is_file($subFile)){ $size+=$this->filesize($subFile); }else{ $size+=$this->getFolderSize($subFile); } } } if($size>0){ $query=OC_DB::prepare("INSERT INTO *PREFIX*foldersize VALUES(?,?)"); $result=$query->execute(array($path,$size)); } } return $size; } /** * @brief clear the folder size cache of folders containing a file * @param string $path */ public function clearFolderSizeCache($path){ if($this->is_file($path)){ $path=dirname($path); } $path=str_replace('//','/',$path); if($this->is_dir($path) and substr($path,-1)!='/'){ $path.='/'; } $query=OC_DB::prepare("DELETE FROM *PREFIX*foldersize WHERE path = ?"); $result=$query->execute(array($path)); if($path!='/' and $path!=''){ $parts=explode('/',$path); //pop empty part $part=array_pop($parts); if(empty($part)){ array_pop($parts); } $parent=implode('/',$parts); $this->clearFolderSizeCache($parent); } } } ?>