* @author Jesús Macias * @author Jörn Friedrich Dreyer * @author Michael Gapczynski * @author Morris Jobke * @author Philipp Kapfer * @author Robin Appelman * @author Robin McCorkell * @author Thomas Müller * @author Vincent Petry * * @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 * */ namespace OCA\Files_External\Lib\Storage; use Icewind\SMB\Change; use Icewind\SMB\Exception\ConnectException; use Icewind\SMB\Exception\Exception; use Icewind\SMB\Exception\ForbiddenException; use Icewind\SMB\Exception\NotFoundException; use Icewind\SMB\INotifyHandler; use Icewind\SMB\IFileInfo; use Icewind\SMB\IShare; use Icewind\SMB\NativeServer; use Icewind\SMB\Server; use Icewind\Streams\CallbackWrapper; use Icewind\Streams\IteratorDirectory; use OC\Cache\CappedMemoryCache; use OC\Files\Filesystem; use OC\Files\Storage\Common; use OCP\Files\Storage\INotifyStorage; use OCP\Files\StorageNotAvailableException; class SMB extends Common implements INotifyStorage { /** * @var \Icewind\SMB\Server */ protected $server; /** * @var \Icewind\SMB\Share */ protected $share; /** * @var string */ protected $root; /** * @var \Icewind\SMB\FileInfo[] */ protected $statCache; public function __construct($params) { if (isset($params['host']) && isset($params['user']) && isset($params['password']) && isset($params['share'])) { if (Server::NativeAvailable()) { $this->server = new NativeServer($params['host'], $params['user'], $params['password']); } else { $this->server = new Server($params['host'], $params['user'], $params['password']); } $this->share = $this->server->getShare(trim($params['share'], '/')); $this->root = isset($params['root']) ? $params['root'] : '/'; if (!$this->root || $this->root[0] != '/') { $this->root = '/' . $this->root; } if (substr($this->root, -1, 1) != '/') { $this->root .= '/'; } } else { throw new \Exception('Invalid configuration'); } $this->statCache = new CappedMemoryCache(); } /** * @return string */ public function getId() { // FIXME: double slash to keep compatible with the old storage ids, // failure to do so will lead to creation of a new storage id and // loss of shares from the storage return 'smb::' . $this->server->getUser() . '@' . $this->server->getHost() . '//' . $this->share->getName() . '/' . $this->root; } /** * @param string $path * @return string */ protected function buildPath($path) { return Filesystem::normalizePath($this->root . '/' . $path, true, false, true); } protected function relativePath($fullPath) { if ($fullPath === $this->root) { return ''; } else if (substr($fullPath, 0, strlen($this->root)) === $this->root) { return substr($fullPath, strlen($this->root)); } else { return null; } } /** * @param string $path * @return \Icewind\SMB\IFileInfo * @throws StorageNotAvailableException */ protected function getFileInfo($path) { try { $path = $this->buildPath($path); if (!isset($this->statCache[$path])) { $this->statCache[$path] = $this->share->stat($path); } return $this->statCache[$path]; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } /** * @param string $path * @return \Icewind\SMB\IFileInfo[] * @throws StorageNotAvailableException */ protected function getFolderContents($path) { try { $path = $this->buildPath($path); $files = $this->share->dir($path); foreach ($files as $file) { $this->statCache[$path . '/' . $file->getName()] = $file; } return array_filter($files, function(IFileInfo $file) { return !$file->isHidden(); }); } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } /** * @param \Icewind\SMB\IFileInfo $info * @return array */ protected function formatInfo($info) { return array( 'size' => $info->getSize(), 'mtime' => $info->getMTime() ); } /** * @param string $path * @return array */ public function stat($path) { $result = $this->formatInfo($this->getFileInfo($path)); if ($this->remoteIsShare() && $this->isRootDir($path)) { $result['mtime'] = $this->shareMTime(); } return $result; } /** * get the best guess for the modification time of the share * * @return int */ private function shareMTime() { $highestMTime = 0; $files = $this->share->dir($this->root); foreach ($files as $fileInfo) { if ($fileInfo->getMTime() > $highestMTime) { $highestMTime = $fileInfo->getMTime(); } } return $highestMTime; } /** * Check if the path is our root dir (not the smb one) * * @param string $path the path * @return bool */ private function isRootDir($path) { return $path === '' || $path === '/' || $path === '.'; } /** * Check if our root points to a smb share * * @return bool true if our root points to a share false otherwise */ private function remoteIsShare() { return $this->share->getName() && (!$this->root || $this->root === '/'); } /** * @param string $path * @return bool */ public function unlink($path) { try { if ($this->is_dir($path)) { return $this->rmdir($path); } else { $path = $this->buildPath($path); unset($this->statCache[$path]); $this->share->del($path); return true; } } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } /** * @param string $path1 the old name * @param string $path2 the new name * @return bool */ public function rename($path1, $path2) { try { $this->remove($path2); $path1 = $this->buildPath($path1); $path2 = $this->buildPath($path2); return $this->share->rename($path1, $path2); } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } /** * check if a file or folder has been updated since $time * * @param string $path * @param int $time * @return bool */ public function hasUpdated($path, $time) { if (!$path and $this->root == '/') { // mtime doesn't work for shares, but giving the nature of the backend, // doing a full update is still just fast enough return true; } else { $actualTime = $this->filemtime($path); return $actualTime > $time; } } /** * @param string $path * @param string $mode * @return resource|false */ public function fopen($path, $mode) { $fullPath = $this->buildPath($path); try { switch ($mode) { case 'r': case 'rb': if (!$this->file_exists($path)) { return false; } return $this->share->read($fullPath); case 'w': case 'wb': $source = $this->share->write($fullPath); return CallBackWrapper::wrap($source, null, null, function () use ($fullPath) { unset($this->statCache[$fullPath]); }); case 'a': case 'ab': case 'r+': case 'w+': case 'wb+': case 'a+': case 'x': case 'x+': case 'c': case 'c+': //emulate these if (strrpos($path, '.') !== false) { $ext = substr($path, strrpos($path, '.')); } else { $ext = ''; } if ($this->file_exists($path)) { if (!$this->isUpdatable($path)) { return false; } $tmpFile = $this->getCachedFile($path); } else { if (!$this->isCreatable(dirname($path))) { return false; } $tmpFile = \OCP\Files::tmpFile($ext); } $source = fopen($tmpFile, $mode); $share = $this->share; return CallbackWrapper::wrap($source, null, null, function () use ($tmpFile, $fullPath, $share) { unset($this->statCache[$fullPath]); $share->put($tmpFile, $fullPath); unlink($tmpFile); }); } return false; } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } public function rmdir($path) { try { $this->statCache = array(); $content = $this->share->dir($this->buildPath($path)); foreach ($content as $file) { if ($file->isDirectory()) { $this->rmdir($path . '/' . $file->getName()); } else { $this->share->del($file->getPath()); } } $this->share->rmdir($this->buildPath($path)); return true; } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } public function touch($path, $time = null) { try { if (!$this->file_exists($path)) { $fh = $this->share->write($this->buildPath($path)); fclose($fh); return true; } return false; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } public function opendir($path) { try { $files = $this->getFolderContents($path); } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } $names = array_map(function ($info) { /** @var \Icewind\SMB\IFileInfo $info */ return $info->getName(); }, $files); return IteratorDirectory::wrap($names); } public function filetype($path) { try { return $this->getFileInfo($path)->isDirectory() ? 'dir' : 'file'; } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } } public function mkdir($path) { $path = $this->buildPath($path); try { $this->share->mkdir($path); return true; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } catch (Exception $e) { return false; } } public function file_exists($path) { try { $this->getFileInfo($path); return true; } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } catch (ConnectException $e) { throw new StorageNotAvailableException($e->getMessage(), $e->getCode(), $e); } } public function isReadable($path) { try { $info = $this->getFileInfo($path); return !$info->isHidden(); } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } } public function isUpdatable($path) { try { $info = $this->getFileInfo($path); // following windows behaviour for read-only folders: they can be written into // (https://support.microsoft.com/en-us/kb/326549 - "cause" section) return !$info->isHidden() && (!$info->isReadOnly() || $this->is_dir($path)); } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } } public function isDeletable($path) { try { $info = $this->getFileInfo($path); return !$info->isHidden() && !$info->isReadOnly(); } catch (NotFoundException $e) { return false; } catch (ForbiddenException $e) { return false; } } /** * check if smbclient is installed */ public static function checkDependencies() { return ( (bool)\OC_Helper::findBinaryPath('smbclient') || Server::NativeAvailable() ) ? true : ['smbclient']; } /** * Test a storage for availability * * @return bool */ public function test() { try { return parent::test(); } catch (Exception $e) { return false; } } public function listen($path, callable $callback) { $fullPath = $this->buildPath($path); $oldRenamePath = null; $this->share->notify($fullPath)->listen(function (Change $change) use (&$oldRenamePath, $callback) { $path = $this->relativePath($change->getPath()); if (is_null($path)) { return true; } if ($change->getCode() === INotifyHandler::NOTIFY_RENAMED_OLD) { $oldRenamePath = $path; return true; } $type = $this->mapNotifyType($change->getCode()); if (is_null($type)) { return true; } if ($type === INotifyStorage::NOTIFY_RENAMED) { if (!is_null($oldRenamePath)) { $result = $callback($type, $oldRenamePath, $path); $oldRenamePath = null; } } else { $result = $callback($type, $path); } return $result; }); } private function mapNotifyType($smbType) { switch ($smbType) { case INotifyHandler::NOTIFY_ADDED: return INotifyStorage::NOTIFY_ADDED; case INotifyHandler::NOTIFY_REMOVED: return INotifyStorage::NOTIFY_REMOVED; case INotifyHandler::NOTIFY_MODIFIED: case INotifyHandler::NOTIFY_ADDED_STREAM: case INotifyHandler::NOTIFY_MODIFIED_STREAM: case INotifyHandler::NOTIFY_REMOVED_STREAM: return INotifyStorage::NOTIFY_MODIFIED; case INotifyHandler::NOTIFY_RENAMED_NEW: return INotifyStorage::NOTIFY_RENAMED; default: return null; } } }