123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489 |
- <?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\Filesystem;
- use OC\Files\Mount\MoveableMount;
- use OC\Files\Utils\PathHelper;
- use OCP\EventDispatcher\GenericEvent;
- use OCP\EventDispatcher\IEventDispatcher;
- use OCP\Files\FileInfo;
- use OCP\Files\InvalidPathException;
- use OCP\Files\IRootFolder;
- use OCP\Files\Node as INode;
- use OCP\Files\NotFoundException;
- use OCP\Files\NotPermittedException;
- use OCP\Lock\LockedException;
- use OCP\PreConditionNotMetException;
- // FIXME: this class really should be abstract (+1)
- class Node implements INode {
- /**
- * @var \OC\Files\View $view
- */
- protected $view;
- protected IRootFolder $root;
- /**
- * @var string $path Absolute path to the node (e.g. /admin/files/folder/file)
- */
- protected $path;
- protected ?FileInfo $fileInfo;
- protected ?INode $parent;
- private bool $infoHasSubMountsIncluded;
- /**
- * @param \OC\Files\View $view
- * @param \OCP\Files\IRootFolder $root
- * @param string $path
- * @param FileInfo $fileInfo
- */
- public function __construct(IRootFolder $root, $view, $path, $fileInfo = null, ?INode $parent = null, bool $infoHasSubMountsIncluded = true) {
- if (Filesystem::normalizePath($view->getRoot()) !== '/') {
- throw new PreConditionNotMetException('The view passed to the node should not have any fake root set');
- }
- $this->view = $view;
- $this->root = $root;
- $this->path = $path;
- $this->fileInfo = $fileInfo;
- $this->parent = $parent;
- $this->infoHasSubMountsIncluded = $infoHasSubMountsIncluded;
- }
- /**
- * Creates a Node of the same type that represents a non-existing path
- *
- * @param string $path path
- * @return Node non-existing node
- * @throws \Exception
- */
- protected function createNonExistingNode($path) {
- throw new \Exception('Must be implemented by subclasses');
- }
- /**
- * Returns the matching file info
- *
- * @return FileInfo
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function getFileInfo(bool $includeMountPoint = true) {
- if (!$this->fileInfo) {
- if (!Filesystem::isValidPath($this->path)) {
- throw new InvalidPathException();
- }
- $fileInfo = $this->view->getFileInfo($this->path, $includeMountPoint);
- $this->infoHasSubMountsIncluded = $includeMountPoint;
- if ($fileInfo instanceof FileInfo) {
- $this->fileInfo = $fileInfo;
- } else {
- throw new NotFoundException();
- }
- } elseif ($includeMountPoint && !$this->infoHasSubMountsIncluded && $this instanceof Folder) {
- if ($this->fileInfo instanceof \OC\Files\FileInfo) {
- $this->view->addSubMounts($this->fileInfo);
- }
- $this->infoHasSubMountsIncluded = true;
- }
- return $this->fileInfo;
- }
- /**
- * @param string[] $hooks
- */
- protected function sendHooks($hooks, ?array $args = null) {
- $args = !empty($args) ? $args : [$this];
- /** @var IEventDispatcher $dispatcher */
- $dispatcher = \OC::$server->get(IEventDispatcher::class);
- foreach ($hooks as $hook) {
- if (method_exists($this->root, 'emit')) {
- $this->root->emit('\OC\Files', $hook, $args);
- }
- if (in_array($hook, ['preWrite', 'postWrite', 'preCreate', 'postCreate', 'preTouch', 'postTouch', 'preDelete', 'postDelete'], true)) {
- $event = new GenericEvent($args[0]);
- } else {
- $event = new GenericEvent($args);
- }
- $dispatcher->dispatch('\OCP\Files::' . $hook, $event);
- }
- }
- /**
- * @param int $permissions
- * @return bool
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- protected function checkPermissions($permissions) {
- return ($this->getPermissions() & $permissions) === $permissions;
- }
- public function delete() {
- }
- /**
- * @param int $mtime
- * @throws InvalidPathException
- * @throws NotFoundException
- * @throws NotPermittedException
- */
- public function touch($mtime = null) {
- if ($this->checkPermissions(\OCP\Constants::PERMISSION_UPDATE)) {
- $this->sendHooks(['preTouch']);
- $this->view->touch($this->path, $mtime);
- $this->sendHooks(['postTouch']);
- if ($this->fileInfo) {
- if (is_null($mtime)) {
- $mtime = time();
- }
- $this->fileInfo['mtime'] = $mtime;
- }
- } else {
- throw new NotPermittedException();
- }
- }
- public function getStorage() {
- $storage = $this->getMountPoint()->getStorage();
- if (!$storage) {
- throw new \Exception('No storage for node');
- }
- return $storage;
- }
- /**
- * @return string
- */
- public function getPath() {
- return $this->path;
- }
- /**
- * @return string
- */
- public function getInternalPath() {
- return $this->getFileInfo(false)->getInternalPath();
- }
- /**
- * @return int
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function getId() {
- return $this->getFileInfo(false)->getId() ?? -1;
- }
- /**
- * @return array
- */
- public function stat() {
- return $this->view->stat($this->path);
- }
- /**
- * @return int
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function getMTime() {
- return $this->getFileInfo()->getMTime();
- }
- /**
- * @param bool $includeMounts
- * @return int|float
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function getSize($includeMounts = true): int|float {
- return $this->getFileInfo()->getSize($includeMounts);
- }
- /**
- * @return string
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function getEtag() {
- return $this->getFileInfo()->getEtag();
- }
- /**
- * @return int
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function getPermissions() {
- return $this->getFileInfo(false)->getPermissions();
- }
- /**
- * @return bool
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function isReadable() {
- return $this->getFileInfo(false)->isReadable();
- }
- /**
- * @return bool
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function isUpdateable() {
- return $this->getFileInfo(false)->isUpdateable();
- }
- /**
- * @return bool
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function isDeletable() {
- return $this->getFileInfo(false)->isDeletable();
- }
- /**
- * @return bool
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function isShareable() {
- return $this->getFileInfo(false)->isShareable();
- }
- /**
- * @return bool
- * @throws InvalidPathException
- * @throws NotFoundException
- */
- public function isCreatable() {
- return $this->getFileInfo(false)->isCreatable();
- }
- public function getParent(): INode|IRootFolder {
- if ($this->parent === null) {
- $newPath = dirname($this->path);
- if ($newPath === '' || $newPath === '.' || $newPath === '/') {
- return $this->root;
- }
- // Manually fetch the parent if the current node doesn't have a file info yet
- try {
- $fileInfo = $this->getFileInfo();
- } catch (NotFoundException) {
- $this->parent = $this->root->get($newPath);
- /** @var \OCP\Files\Folder $this->parent */
- return $this->parent;
- }
- // gather the metadata we already know about our parent
- $parentData = [
- 'path' => $newPath,
- 'fileid' => $fileInfo->getParentId(),
- ];
- // and create lazy folder with it instead of always querying
- $this->parent = new LazyFolder($this->root, function () use ($newPath) {
- return $this->root->get($newPath);
- }, $parentData);
- }
- return $this->parent;
- }
- /**
- * @return string
- */
- public function getName() {
- return basename($this->path);
- }
- /**
- * @param string $path
- * @return string
- */
- protected function normalizePath($path) {
- return PathHelper::normalizePath($path);
- }
- /**
- * check if the requested path is valid
- *
- * @param string $path
- * @return bool
- */
- public function isValidPath($path) {
- return Filesystem::isValidPath($path);
- }
- public function isMounted() {
- return $this->getFileInfo(false)->isMounted();
- }
- public function isShared() {
- return $this->getFileInfo(false)->isShared();
- }
- public function getMimeType() {
- return $this->getFileInfo(false)->getMimetype();
- }
- public function getMimePart() {
- return $this->getFileInfo(false)->getMimePart();
- }
- public function getType() {
- return $this->getFileInfo(false)->getType();
- }
- public function isEncrypted() {
- return $this->getFileInfo(false)->isEncrypted();
- }
- public function getMountPoint() {
- return $this->getFileInfo(false)->getMountPoint();
- }
- public function getOwner() {
- return $this->getFileInfo(false)->getOwner();
- }
- public function getChecksum() {
- }
- public function getExtension(): string {
- return $this->getFileInfo(false)->getExtension();
- }
- /**
- * @param int $type \OCP\Lock\ILockingProvider::LOCK_SHARED or \OCP\Lock\ILockingProvider::LOCK_EXCLUSIVE
- * @throws LockedException
- */
- public function lock($type) {
- $this->view->lockFile($this->path, $type);
- }
- /**
- * @param int $type \OCP\Lock\ILockingProvider::LOCK_SHARED or \OCP\Lock\ILockingProvider::LOCK_EXCLUSIVE
- * @throws LockedException
- */
- public function changeLock($type) {
- $this->view->changeLock($this->path, $type);
- }
- /**
- * @param int $type \OCP\Lock\ILockingProvider::LOCK_SHARED or \OCP\Lock\ILockingProvider::LOCK_EXCLUSIVE
- * @throws LockedException
- */
- public function unlock($type) {
- $this->view->unlockFile($this->path, $type);
- }
- /**
- * @param string $targetPath
- * @return INode
- * @throws InvalidPathException
- * @throws NotFoundException
- * @throws NotPermittedException if copy not allowed or failed
- */
- public function copy($targetPath) {
- $targetPath = $this->normalizePath($targetPath);
- $parent = $this->root->get(dirname($targetPath));
- if ($parent instanceof Folder and $this->isValidPath($targetPath) and $parent->isCreatable()) {
- $nonExisting = $this->createNonExistingNode($targetPath);
- $this->sendHooks(['preCopy'], [$this, $nonExisting]);
- $this->sendHooks(['preWrite'], [$nonExisting]);
- if (!$this->view->copy($this->path, $targetPath)) {
- throw new NotPermittedException('Could not copy ' . $this->path . ' to ' . $targetPath);
- }
- $targetNode = $this->root->get($targetPath);
- $this->sendHooks(['postCopy'], [$this, $targetNode]);
- $this->sendHooks(['postWrite'], [$targetNode]);
- return $targetNode;
- } else {
- throw new NotPermittedException('No permission to copy to path ' . $targetPath);
- }
- }
- /**
- * @param string $targetPath
- * @return INode
- * @throws InvalidPathException
- * @throws NotFoundException
- * @throws NotPermittedException if move not allowed or failed
- * @throws LockedException
- */
- public function move($targetPath) {
- $targetPath = $this->normalizePath($targetPath);
- $parent = $this->root->get(dirname($targetPath));
- if (
- ($parent instanceof Folder)
- && $this->isValidPath($targetPath)
- && (
- $parent->isCreatable()
- || (
- $parent->getInternalPath() === ''
- && ($parent->getMountPoint() instanceof MoveableMount)
- )
- )
- ) {
- $nonExisting = $this->createNonExistingNode($targetPath);
- $this->sendHooks(['preRename'], [$this, $nonExisting]);
- $this->sendHooks(['preWrite'], [$nonExisting]);
- if (!$this->view->rename($this->path, $targetPath)) {
- throw new NotPermittedException('Could not move ' . $this->path . ' to ' . $targetPath);
- }
- $mountPoint = $this->getMountPoint();
- if ($mountPoint) {
- // update the cached fileinfo with the new (internal) path
- /** @var \OC\Files\FileInfo $oldFileInfo */
- $oldFileInfo = $this->getFileInfo();
- $this->fileInfo = new \OC\Files\FileInfo($targetPath, $oldFileInfo->getStorage(), $mountPoint->getInternalPath($targetPath), $oldFileInfo->getData(), $mountPoint, $oldFileInfo->getOwner());
- }
- $targetNode = $this->root->get($targetPath);
- $this->sendHooks(['postRename'], [$this, $targetNode]);
- $this->sendHooks(['postWrite'], [$targetNode]);
- $this->path = $targetPath;
- return $targetNode;
- } else {
- throw new NotPermittedException('No permission to move to path ' . $targetPath);
- }
- }
- public function getCreationTime(): int {
- return $this->getFileInfo()->getCreationTime();
- }
- public function getUploadTime(): int {
- return $this->getFileInfo()->getUploadTime();
- }
- public function getParentId(): int {
- return $this->fileInfo->getParentId();
- }
- /**
- * @inheritDoc
- * @return array<string, int|string|bool|float|string[]|int[]>
- */
- public function getMetadata(): array {
- return $this->fileInfo->getMetadata();
- }
- }
|