123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530 |
- <?php
- /**
- * SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
- namespace OC\Files\Node;
- use OC\Files\FileInfo;
- use OC\Files\Mount\Manager;
- use OC\Files\Mount\MountPoint;
- use OC\Files\Utils\PathHelper;
- use OC\Files\View;
- use OC\Hooks\PublicEmitter;
- use OC\User\NoUserException;
- use OCP\Cache\CappedMemoryCache;
- use OCP\EventDispatcher\IEventDispatcher;
- use OCP\Files\Cache\ICacheEntry;
- use OCP\Files\Config\IUserMountCache;
- use OCP\Files\Events\Node\FilesystemTornDownEvent;
- use OCP\Files\IRootFolder;
- use OCP\Files\Mount\IMountPoint;
- use OCP\Files\Node as INode;
- use OCP\Files\NotFoundException;
- use OCP\Files\NotPermittedException;
- use OCP\ICache;
- use OCP\ICacheFactory;
- use OCP\IUser;
- use OCP\IUserManager;
- use OCP\Server;
- use Psr\Log\LoggerInterface;
- /**
- * Class Root
- *
- * Hooks available in scope \OC\Files
- * - preWrite(\OCP\Files\Node $node)
- * - postWrite(\OCP\Files\Node $node)
- * - preCreate(\OCP\Files\Node $node)
- * - postCreate(\OCP\Files\Node $node)
- * - preDelete(\OCP\Files\Node $node)
- * - postDelete(\OCP\Files\Node $node)
- * - preTouch(\OC\FilesP\Node $node, int $mtime)
- * - postTouch(\OCP\Files\Node $node)
- * - preCopy(\OCP\Files\Node $source, \OCP\Files\Node $target)
- * - postCopy(\OCP\Files\Node $source, \OCP\Files\Node $target)
- * - preRename(\OCP\Files\Node $source, \OCP\Files\Node $target)
- * - postRename(\OCP\Files\Node $source, \OCP\Files\Node $target)
- *
- * @package OC\Files\Node
- */
- class Root extends Folder implements IRootFolder {
- private Manager $mountManager;
- private PublicEmitter $emitter;
- private ?IUser $user;
- private CappedMemoryCache $userFolderCache;
- private IUserMountCache $userMountCache;
- private LoggerInterface $logger;
- private IUserManager $userManager;
- private IEventDispatcher $eventDispatcher;
- private ICache $pathByIdCache;
- /**
- * @param Manager $manager
- * @param View $view
- * @param IUser|null $user
- */
- public function __construct(
- $manager,
- $view,
- $user,
- IUserMountCache $userMountCache,
- LoggerInterface $logger,
- IUserManager $userManager,
- IEventDispatcher $eventDispatcher,
- ICacheFactory $cacheFactory,
- ) {
- parent::__construct($this, $view, '');
- $this->mountManager = $manager;
- $this->user = $user;
- $this->emitter = new PublicEmitter();
- $this->userFolderCache = new CappedMemoryCache();
- $this->userMountCache = $userMountCache;
- $this->logger = $logger;
- $this->userManager = $userManager;
- $eventDispatcher->addListener(FilesystemTornDownEvent::class, function () {
- $this->userFolderCache = new CappedMemoryCache();
- });
- $this->pathByIdCache = $cacheFactory->createLocal('path-by-id');
- }
- /**
- * Get the user for which the filesystem is setup
- *
- * @return \OC\User\User
- */
- public function getUser() {
- return $this->user;
- }
- /**
- * @param string $scope
- * @param string $method
- * @param callable $callback
- */
- public function listen($scope, $method, callable $callback) {
- $this->emitter->listen($scope, $method, $callback);
- }
- /**
- * @param string $scope optional
- * @param string $method optional
- * @param callable $callback optional
- */
- public function removeListener($scope = null, $method = null, ?callable $callback = null) {
- $this->emitter->removeListener($scope, $method, $callback);
- }
- /**
- * @param string $scope
- * @param string $method
- * @param Node[] $arguments
- */
- public function emit($scope, $method, $arguments = []) {
- $this->emitter->emit($scope, $method, $arguments);
- }
- /**
- * @param \OC\Files\Storage\Storage $storage
- * @param string $mountPoint
- * @param array $arguments
- */
- public function mount($storage, $mountPoint, $arguments = []) {
- $mount = new MountPoint($storage, $mountPoint, $arguments);
- $this->mountManager->addMount($mount);
- }
- public function getMount(string $mountPoint): IMountPoint {
- return $this->mountManager->find($mountPoint);
- }
- /**
- * @param string $mountPoint
- * @return \OC\Files\Mount\MountPoint[]
- */
- public function getMountsIn(string $mountPoint): array {
- return $this->mountManager->findIn($mountPoint);
- }
- /**
- * @param string $storageId
- * @return \OC\Files\Mount\MountPoint[]
- */
- public function getMountByStorageId($storageId) {
- return $this->mountManager->findByStorageId($storageId);
- }
- /**
- * @param int $numericId
- * @return MountPoint[]
- */
- public function getMountByNumericStorageId($numericId) {
- return $this->mountManager->findByNumericId($numericId);
- }
- /**
- * @param \OC\Files\Mount\MountPoint $mount
- */
- public function unMount($mount) {
- $this->mountManager->remove($mount);
- }
- public function get($path) {
- $path = $this->normalizePath($path);
- if ($this->isValidPath($path)) {
- $fullPath = $this->getFullPath($path);
- $fileInfo = $this->view->getFileInfo($fullPath, false);
- if ($fileInfo) {
- return $this->createNode($fullPath, $fileInfo, false);
- } else {
- throw new NotFoundException($path);
- }
- } else {
- throw new NotPermittedException();
- }
- }
- //most operations can't be done on the root
- /**
- * @param string $targetPath
- * @return Node
- * @throws \OCP\Files\NotPermittedException
- */
- public function rename($targetPath) {
- throw new NotPermittedException();
- }
- public function delete() {
- throw new NotPermittedException();
- }
- /**
- * @param string $targetPath
- * @return Node
- * @throws \OCP\Files\NotPermittedException
- */
- public function copy($targetPath) {
- throw new NotPermittedException();
- }
- /**
- * @param int $mtime
- * @throws \OCP\Files\NotPermittedException
- */
- public function touch($mtime = null) {
- throw new NotPermittedException();
- }
- /**
- * @return \OC\Files\Storage\Storage
- * @throws \OCP\Files\NotFoundException
- */
- public function getStorage() {
- throw new NotFoundException();
- }
- /**
- * @return string
- */
- public function getPath() {
- return '/';
- }
- /**
- * @return string
- */
- public function getInternalPath() {
- return '';
- }
- /**
- * @return int
- */
- public function getId() {
- return 0;
- }
- /**
- * @return array
- */
- public function stat() {
- return [];
- }
- /**
- * @return int
- */
- public function getMTime() {
- return 0;
- }
- /**
- * @param bool $includeMounts
- * @return int|float
- */
- public function getSize($includeMounts = true): int|float {
- return 0;
- }
- /**
- * @return string
- */
- public function getEtag() {
- return '';
- }
- /**
- * @return int
- */
- public function getPermissions() {
- return \OCP\Constants::PERMISSION_CREATE;
- }
- /**
- * @return bool
- */
- public function isReadable() {
- return false;
- }
- /**
- * @return bool
- */
- public function isUpdateable() {
- return false;
- }
- /**
- * @return bool
- */
- public function isDeletable() {
- return false;
- }
- /**
- * @return bool
- */
- public function isShareable() {
- return false;
- }
- /**
- * @throws \OCP\Files\NotFoundException
- */
- public function getParent(): INode|IRootFolder {
- throw new NotFoundException();
- }
- /**
- * @return string
- */
- public function getName() {
- return '';
- }
- /**
- * Returns a view to user's files folder
- *
- * @param string $userId user ID
- * @return \OCP\Files\Folder
- * @throws NoUserException
- * @throws NotPermittedException
- */
- public function getUserFolder($userId) {
- $userObject = $this->userManager->get($userId);
- if (is_null($userObject)) {
- $e = new NoUserException('Backends provided no user object');
- $this->logger->error(
- sprintf(
- 'Backends provided no user object for %s',
- $userId
- ),
- [
- 'app' => 'files',
- 'exception' => $e,
- ]
- );
- throw $e;
- }
- $userId = $userObject->getUID();
- if (!$this->userFolderCache->hasKey($userId)) {
- if ($this->mountManager->getSetupManager()->isSetupComplete($userObject)) {
- try {
- $folder = $this->get('/' . $userId . '/files');
- if (!$folder instanceof \OCP\Files\Folder) {
- throw new \Exception("Account folder for \"$userId\" exists as a file");
- }
- } catch (NotFoundException $e) {
- if (!$this->nodeExists('/' . $userId)) {
- $this->newFolder('/' . $userId);
- }
- $folder = $this->newFolder('/' . $userId . '/files');
- }
- } else {
- $folder = new LazyUserFolder($this, $userObject, $this->mountManager);
- }
- $this->userFolderCache->set($userId, $folder);
- }
- return $this->userFolderCache->get($userId);
- }
- public function getUserMountCache() {
- return $this->userMountCache;
- }
- public function getFirstNodeByIdInPath(int $id, string $path): ?INode {
- // scope the cache by user, so we don't return nodes for different users
- if ($this->user) {
- $cachedPath = $this->pathByIdCache->get($this->user->getUID() . '::' . $id);
- if ($cachedPath && str_starts_with($path, $cachedPath)) {
- // getting the node by path is significantly cheaper than finding it by id
- $node = $this->get($cachedPath);
- // by validating that the cached path still has the requested fileid we can work around the need to invalidate the cached path
- // if the cached path is invalid or a different file now we fall back to the uncached logic
- if ($node && $node->getId() === $id) {
- return $node;
- }
- }
- }
- $node = current($this->getByIdInPath($id, $path));
- if (!$node) {
- return null;
- }
- if ($this->user) {
- $this->pathByIdCache->set($this->user->getUID() . '::' . $id, $node->getPath());
- }
- return $node;
- }
- /**
- * @param int $id
- * @return Node[]
- */
- public function getByIdInPath(int $id, string $path): array {
- $mountCache = $this->getUserMountCache();
- if (strpos($path, '/', 1) > 0) {
- [, $user] = explode('/', $path);
- } else {
- $user = null;
- }
- $mountsContainingFile = $mountCache->getMountsForFileId($id, $user);
- // if the mount isn't in the cache yet, perform a setup first, then try again
- if (count($mountsContainingFile) === 0) {
- $this->mountManager->getSetupManager()->setupForPath($path, true);
- $mountsContainingFile = $mountCache->getMountsForFileId($id, $user);
- }
- // when a user has access through the same storage through multiple paths
- // (such as an external storage that is both mounted for a user and shared to the user)
- // the mount cache will only hold a single entry for the storage
- // this can lead to issues as the different ways the user has access to a storage can have different permissions
- //
- // so instead of using the cached entries directly, we instead filter the current mounts by the rootid of the cache entry
- $mountRootIds = array_map(function ($mount) {
- return $mount->getRootId();
- }, $mountsContainingFile);
- $mountRootPaths = array_map(function ($mount) {
- return $mount->getRootInternalPath();
- }, $mountsContainingFile);
- $mountProviders = array_unique(array_map(function ($mount) {
- return $mount->getMountProvider();
- }, $mountsContainingFile));
- $mountRoots = array_combine($mountRootIds, $mountRootPaths);
- $mounts = $this->mountManager->getMountsByMountProvider($path, $mountProviders);
- $mountsContainingFile = array_filter($mounts, function ($mount) use ($mountRoots) {
- return isset($mountRoots[$mount->getStorageRootId()]);
- });
- if (count($mountsContainingFile) === 0) {
- if ($user === $this->getAppDataDirectoryName()) {
- $folder = $this->get($path);
- if ($folder instanceof Folder) {
- return $folder->getByIdInRootMount($id);
- } else {
- throw new \Exception('getByIdInPath with non folder');
- }
- }
- return [];
- }
- $nodes = array_map(function (IMountPoint $mount) use ($id, $mountRoots) {
- $rootInternalPath = $mountRoots[$mount->getStorageRootId()];
- $cacheEntry = $mount->getStorage()->getCache()->get($id);
- if (!$cacheEntry) {
- return null;
- }
- // cache jails will hide the "true" internal path
- $internalPath = ltrim($rootInternalPath . '/' . $cacheEntry->getPath(), '/');
- $pathRelativeToMount = substr($internalPath, strlen($rootInternalPath));
- $pathRelativeToMount = ltrim($pathRelativeToMount, '/');
- $absolutePath = rtrim($mount->getMountPoint() . $pathRelativeToMount, '/');
- $storage = $mount->getStorage();
- if ($storage === null) {
- return null;
- }
- $ownerId = $storage->getOwner($pathRelativeToMount);
- if ($ownerId !== false) {
- $owner = Server::get(IUserManager::class)->get($ownerId);
- } else {
- $owner = null;
- }
- return $this->createNode($absolutePath, new FileInfo(
- $absolutePath,
- $storage,
- $cacheEntry->getPath(),
- $cacheEntry,
- $mount,
- $owner,
- ));
- }, $mountsContainingFile);
- $nodes = array_filter($nodes);
- $folders = array_filter($nodes, function (Node $node) use ($path) {
- return PathHelper::getRelativePath($path, $node->getPath()) !== null;
- });
- usort($folders, function ($a, $b) {
- return $b->getPath() <=> $a->getPath();
- });
- return $folders;
- }
- public function getNodeFromCacheEntryAndMount(ICacheEntry $cacheEntry, IMountPoint $mountPoint): INode {
- $path = $cacheEntry->getPath();
- $fullPath = $mountPoint->getMountPoint() . $path;
- // todo: LazyNode?
- $info = new FileInfo($fullPath, $mountPoint->getStorage(), $path, $cacheEntry, $mountPoint);
- $parentPath = dirname($fullPath);
- $parent = new LazyFolder($this, function () use ($parentPath) {
- $parent = $this->get($parentPath);
- if ($parent instanceof \OCP\Files\Folder) {
- return $parent;
- } else {
- throw new \Exception("parent $parentPath is not a folder");
- }
- }, [
- 'path' => $parentPath,
- ]);
- $isDir = $info->getType() === FileInfo::TYPE_FOLDER;
- $view = new View('');
- if ($isDir) {
- return new Folder($this, $view, $path, $info, $parent);
- } else {
- return new File($this, $view, $path, $info, $parent);
- }
- }
- }
|