. * */ require_once( 'lib_share.php' ); /** * Convert target path to source path and pass the function call to the correct storage provider */ class OC_Filestorage_Shared extends OC_Filestorage { private $datadir; private $sourcePaths = array(); public function __construct($arguments) { $this->datadir = $arguments['datadir']; if (!OC_Filesystem::is_dir($this->datadir)) { OC_Filesystem::mkdir($this->datadir); } $this->datadir .= "/"; } public function getInternalPath($path) { $mountPoint = OC_Filesystem::getMountPoint($path); $internalPath = substr($path, strlen($mountPoint)); return $internalPath; } public function getSource($target) { $target = $this->datadir.$target; if (array_key_exists($target, $this->sourcePaths)) { return $this->sourcePaths[$target]; } else { $source = OC_Share::getSource($target); $this->sourcePaths[$target] = $source; return $source; } } public function mkdir($path) { if ($path == "" || $path == "/") { return false; } else { $source = $this->getSource($path); if ($source) { if ($this->is_writeable($path)) { $storage = OC_Filesystem::getStorage($source); return $storage->mkdir($this->getInternalPath($source)); } } } } public function rmdir($path) { // The folder will be removed from the database, but won't be deleted from the owner's filesystem OC_Share::unshareFromMySelf($this->datadir.$path); $this->clearFolderSizeCache($path); } public function opendir($path) { if ($path == "" || $path == "/") { global $FAKEDIRS; $path = $this->datadir.$path; $sharedItems = OC_Share::getItemsInFolder($path); if (empty($sharedItems)) { return false; } $files = array(); foreach ($sharedItems as $item) { // If item is in the root of the shared storage provider add it to the fakedirs if (dirname($item['target'])."/" == $path) { $files[] = basename($item['target']); } } $FAKEDIRS['shared'] = $files; return opendir('fakedir://shared'); } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); $dh = $storage->opendir($this->getInternalPath($source)); // Remove any duplicate or trailing '/' $path = rtrim($this->datadir.$path, "/"); $path = preg_replace('{(/)\1+}', "/", $path); $modifiedItems = OC_Share::getItemsInFolder($source); if ($modifiedItems && $dh) { global $FAKEDIRS; $sources = array(); $targets = array(); foreach ($modifiedItems as $item) { // If the item is in the current directory and has a different name than the source, add it to the arrays if (dirname($item['target']) == $path && basename($item['source']) != basename($item['target'])) { $sources[] = basename($item['source']); $targets[] = basename($item['target']); // If the item was unshared from self, add it it to the arrays } elseif ($item['target'] == "/") { $sources[] = basename($item['source']); $targets[] = ""; } } // Don't waste time if there aren't any modified items in the current directory if (empty($sources)) { return $dh; } else { while (($filename = readdir($dh)) !== false) { if ($filename != "." && $filename != "..") { // If the file isn't in the sources array it isn't modified and can be added as is if (!in_array($filename, $sources)) { $files[] = $filename; // The file has a different name than the source and is added to the fakedirs } else { $target = $targets[array_search($filename, $sources)]; // Don't add the file if it was unshared from self by the user if ($target != "") { $files[] = $target; } } } } $FAKEDIRS['shared'] = $files; return opendir('fakedir://shared'); } } else { return $dh; } } } } public function is_dir($path) { if ($path == "" || $path == "/") { return true; } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->is_dir($this->getInternalPath($source)); } } } public function is_file($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->is_file($this->getInternalPath($source)); } } // TODO fill in other components of array public function stat($path) { if ($path == "" || $path == "/") { $stat["dev"] = ""; $stat["ino"] = ""; $stat["mode"] = ""; $stat["nlink"] = ""; $stat["uid"] = ""; $stat["gid"] = ""; $stat["rdev"] = ""; $stat["size"] = $this->filesize($path); $stat["atime"] = $this->fileatime($path); $stat["mtime"] = $this->filemtime($path); $stat["ctime"] = $this->filectime($path); $stat["blksize"] = ""; $stat["blocks"] = ""; return $stat; } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->stat($this->getInternalPath($source)); } } } public function filetype($path) { if ($path == "" || $path == "/") { return "dir"; } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->filetype($this->getInternalPath($source)); } } } public function filesize($path) { if ($path == "" || $path == "/" || $this->is_dir($path)) { return $this->getFolderSize($path); } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->filesize($this->getInternalPath($source)); } } } public function getFolderSize($path) { $dbpath = OC_User::getUser()."/files/Share/".$path."/"; $query = OC_DB::prepare("SELECT size FROM *PREFIX*foldersize WHERE path=?"); $size = $query->execute(array($dbpath))->fetchAll(); if (count($size) > 0) { return $size[0]['size']; } else { return $this->calculateFolderSize($path); } } public function calculateFolderSize($path) { if ($this->is_file($path)) { $path = dirname($path); } $path = str_replace("//", "/", $path); if ($this->is_dir($path) && substr($path, -1) != "/") { $path .= "/"; } $size = 0; if ($dh = $this->opendir($path)) { while (($filename = readdir($dh)) !== false) { if ($filename != "." && $filename != "..") { $subFile = $path.$filename; if ($this->is_file($subFile)) { $size += $this->filesize($subFile); } else { $size += $this->getFolderSize($subFile); } } } if ($size > 0) { $dbpath = OC_User::getUser()."/files/Share/".$path; $query = OC_DB::prepare("INSERT INTO *PREFIX*foldersize VALUES(?,?)"); $result = $query->execute(array($dbpath, $size)); } } return $size; } public function clearFolderSizeCache($path){ if ($this->is_file($path)) { $path = dirname($path); } $path = str_replace("//", "/", $path); if ($this->is_dir($path) && substr($path, -1) != "/") { $path .= "/"; } $query = OC_DB::prepare("DELETE FROM *PREFIX*foldersize WHERE path = ?"); $result = $query->execute(array($path)); if ($path != "/" && $path != "") { $parts = explode("/", $path); $part = array_pop($parts); if (empty($part)) { array_pop($parts); } $parent = implode("/", $parts); $this->clearFolderSizeCache($parent); } } public function is_readable($path) { return true; } public function is_writeable($path) { if ($path == "" || $path == "/" || OC_Share::getPermissions($this->datadir.$path) & OC_Share::WRITE) { return true; } else { return false; } } public function file_exists($path) { if ($path == "" || $path == "/") { return true; } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->file_exists($this->getInternalPath($source)); } } } public function readfile($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->readfile($this->getInternalPath($source)); } } public function filectime($path) { if ($path == "" || $path == "/") { $ctime = 0; if ($dh = $this->opendir($path)) { while (($filename = readdir($dh)) !== false) { $tempctime = $this->filectime($filename); if ($tempctime < $ctime) { $ctime = $tempctime; } } return $ctime; } } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->filectime($this->getInternalPath($source)); } } } public function filemtime($path) { if ($path == "" || $path == "/") { $mtime = 0; if ($dh = $this->opendir($path)) { while (($filename = readdir($dh)) !== false) { $tempmtime = $this->filemtime($filename); if ($tempmtime > $mtime) { $mtime = $tempmtime; } } return $mtime; } } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->filemtime($this->getInternalPath($source)); } } } public function fileatime($path) { if ($path == "" || $path == "/") { $atime = 0; if ($dh = $this->opendir($path)) { while (($filename = readdir($dh)) !== false) { $tempatime = $this->fileatime($filename); if ($tempatime > $atime) { $atime = $tempatime; } } return $atime; } } else { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->fileatime($this->getInternalPath($source)); } } } public function file_get_contents($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->file_get_contents($this->getInternalPath($source)); } } public function file_put_contents($path, $data) { if ($this->is_writeable($path)) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->file_put_contents($this->getInternalPath($source), $data); } } } public function unlink($path) { $target = $this->datadir.$path; // If the user has delete permission for the item, the source item will be deleted if (OC_Share::getPermissions($target) & OC_Share::DELETE) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->unlink($this->getInternalPath($source)); } // The item will be removed from the database, but won't be touched on the owner's filesystem } else { // Check if the item is inside a shared folder if (OC_Share::getParentFolders($target)) { // If entry for item already exists if (OC_Share::getItem($target)) { OC_Share::setTarget($target, "/"); } else { OC_Share::pullOutOfFolder($target, "/"); } // Delete the database entry } else { OC_Share::unshareFromMySelf($target); } $this->clearFolderSizeCache($this->getInternalPath($target)); } return true; } public function rename($path1, $path2) { // If the user has write permission for the item, the source item will be renamed if ($this->is_writeable($path1)) { $source = $this->getSource($path1); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->rename($path1, $path2); } // The item will be renamed in the database, but won't be touched on the owner's filesystem } else { $oldTarget = $this->datadir.$path1; $newTarget = $this->datadir.$path2; if (OC_Share::getItem($oldTarget)) { OC_Share::setTarget($oldTarget, $newTarget); // There is no entry in the database for the item, it must be inside a shared folder } else { OC_Share::pullOutOfFolder($oldTarget, $newTarget); // If this is a folder being renamed, call setTarget in case there are any database entries inside the folder if (self::is_dir($path1)) { OC_Share::setTarget($oldTarget, $newTarget); } } $this->clearFolderSizeCache($this->getInternalPath($oldTarget)); $this->clearFolderSizeCache($this->getInternalPath($newTarget)); return true; } } public function copy($path1, $path2) { if ($path2 == "" || $path2 == "/") { // TODO Construct new shared item or should this not be allowed? } else { if ($this->is_writeable($path2)) { $tmpFile = $this->toTmpFile($path1); return $this->fromTmpFile($tmpFile, $path2); } else { return false; } } } public function fopen($path, $mode) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->fopen($this->getInternalPath($source), $mode); } } public function toTmpFile($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->toTmpFile($this->getInternalPath($source)); } } public function fromTmpFile($tmpFile, $path) { if ($this->is_writeable($path)) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->fromTmpFile($tmpFile, $this->getInternalPath($source)); } } else { return false; } } public function fromUploadedFile($tmpPath, $path) { $source = $this->getSource($tmpPath); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->fromUploadedFile($this->getInternalPath($source), $path); } } public function getMimeType($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->getMimeType($this->getInternalPath($source)); } } public function delTree($path) { $target = $this->datadir.$path; if (OC_Share::getPermissions($target) & OC_Share::DELETE) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->delTree($this->getInternalPath($source)); } } else { // Check if the folder is inside a shared folder if (OC_Share::getParentFolders($target)) { // If entry for folder already exists if (OC_Share::getItem($target)) { OC_Share::setTarget($target, "/"); } else { OC_Share::pullOutOfFolder($target, "/"); // Call setTarget in case there are any database entries for items inside this folder OC_Share::setTarget($target, "/"); } // Delete the database entry } else { OC_Share::unshareFromMySelf($target); } $this->clearFolderSizeCache($this->getInternalPath($target)); return true; } } public function find($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->find($this->getInternalPath($source)); } } public function getTree($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->getTree($this->getInternalPath($source)); } } public function hash($type, $path, $raw) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->hash($type, $this->getInternalPath($source), $raw); } } public function free_space($path) { $source = $this->getSource($path); if ($source) { $storage = OC_Filesystem::getStorage($source); return $storage->free_space($this->getInternalPath($source)); } } // TODO query all shared files? public function search($query) { } } ?>