owncloud/core

View on GitHub
lib/private/Files/Node/Node.php

Summary

Maintainability
B
5 hrs
Test Coverage
<?php
/**
 * @author Bernhard Posselt <dev@bernhard-posselt.com>
 * @author Joas Schilling <coding@schilljs.com>
 * @author Morris Jobke <hey@morrisjobke.de>
 * @author Robin Appelman <icewind@owncloud.com>
 * @author Roeland Jago Douma <rullzer@owncloud.com>
 * @author Thomas Müller <thomas.mueller@tmit.eu>
 * @author Vincent Petry <pvince81@owncloud.com>
 *
 * @copyright Copyright (c) 2018, ownCloud GmbH
 * @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 <http://www.gnu.org/licenses/>
 *
 */

namespace OC\Files\Node;

use OC\Files\Filesystem;
use OCP\Files\FileInfo;
use OCP\Files\InvalidPathException;
use OCP\Files\NotFoundException;
use OCP\Files\NotPermittedException;

// FIXME: this class really should be abstract
class Node implements \OCP\Files\Node {
    /**
     * @var \OC\Files\View $view
     */
    protected $view;

    /**
     * @var \OC\Files\Node\Root $root
     */
    protected $root;

    /**
     * @var string $path
     */
    protected $path;

    /**
     * @var \OCP\Files\FileInfo
     */
    protected $fileInfo;

    /**
     * @param \OC\Files\View $view
     * @param \OC\Files\Node\Root $root
     * @param string $path
     * @param FileInfo $fileInfo
     */
    public function __construct($root, $view, $path, $fileInfo = null) {
        $this->view = $view;
        $this->root = $root;
        $this->path = $path;
        $this->fileInfo = $fileInfo;
    }

    /**
     * Creates a Node of the same type that represents a nonexistent path
     *
     * @param string $path path
     * @return string nonexistent node class
     */
    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() {
        if (!Filesystem::isValidPath($this->path)) {
            throw new InvalidPathException();
        }
        if (!$this->fileInfo) {
            $fileInfo = $this->view->getFileInfo($this->path);
            if ($fileInfo instanceof FileInfo) {
                $this->fileInfo = $fileInfo;
            } else {
                throw new NotFoundException();
            }
        }
        return $this->fileInfo;
    }

    /**
     * @param string[] $hooks
     */
    protected function sendHooks($hooks) {
        foreach ($hooks as $hook) {
            $this->root->emit('\OC\Files', $hook, [$this]);
        }
    }

    /**
     * @param int $permissions
     * @return bool
     */
    protected function checkPermissions($permissions) {
        return ($this->getPermissions() & $permissions) === $permissions;
    }

    public function delete() {
        return;
    }

    /**
     * @param int $mtime
     * @throws \OCP\Files\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 ($mtime === null) {
                    $mtime = \time();
                }
                $this->fileInfo['mtime'] = $mtime;
            }
        } else {
            throw new NotPermittedException();
        }
    }

    /**
     * @return \OC\Files\Storage\Storage
     * @throws \OCP\Files\NotFoundException
     */
    public function getStorage() {
        list($storage, ) = $this->view->resolvePath($this->path);
        return $storage;
    }

    /**
     * @return string
     */
    public function getPath() {
        return $this->path;
    }

    /**
     * @return string
     */
    public function getInternalPath() {
        list(, $internalPath) = $this->view->resolvePath($this->path);
        return $internalPath;
    }

    /**
     * @return int
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function getId() {
        return $this->getFileInfo()->getId();
    }

    /**
     * @return array
     */
    public function stat() {
        return $this->view->stat($this->path);
    }

    /**
     * @return int
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function getMTime() {
        return $this->getFileInfo()->getMTime();
    }

    /**
     * @return int|float
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function getSize() {
        return $this->getFileInfo()->getSize();
    }

    /**
     * @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()->getPermissions();
    }

    /**
     * @return bool
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function isReadable() {
        return $this->getFileInfo()->isReadable();
    }

    /**
     * @return bool
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function isUpdateable() {
        return $this->getFileInfo()->isUpdateable();
    }

    /**
     * @return bool
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function isDeletable() {
        return $this->getFileInfo()->isDeletable();
    }

    /**
     * @return bool
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function isShareable() {
        return $this->getFileInfo()->isShareable();
    }

    /**
     * @return bool
     * @throws InvalidPathException
     * @throws NotFoundException
     */
    public function isCreatable() {
        return $this->getFileInfo()->isCreatable();
    }

    /**
     * @return Node
     */
    public function getParent() {
        return $this->root->get(\dirname($this->path));
    }

    /**
     * @return string
     */
    public function getName() {
        return \basename($this->path);
    }

    /**
     * @param string $path
     * @return string
     */
    protected function normalizePath($path) {
        if ($path === '' or $path === '/') {
            return '/';
        }
        //no windows style slashes
        $path = \str_replace('\\', '/', $path);
        //add leading slash
        if ($path[0] !== '/') {
            $path = '/' . $path;
        }
        //remove duplicate slashes
        while (\strpos($path, '//') !== false) {
            $path = \str_replace('//', '/', $path);
        }
        //remove trailing slash
        $path = \rtrim($path, '/');

        return $path;
    }

    /**
     * check if the requested path is valid
     *
     * @param string $path
     * @return bool
     */
    public function isValidPath($path) {
        if (!$path || $path[0] !== '/') {
            $path = '/' . $path;
        }
        if (\strstr($path, '/../') || \strrchr($path, '/') === '/..') {
            return false;
        }
        return true;
    }

    public function isMounted() {
        return $this->getFileInfo()->isMounted();
    }

    public function isShared() {
        return $this->getFileInfo()->isShared();
    }

    public function getMimeType() {
        return $this->getFileInfo()->getMimetype();
    }

    public function getMimePart() {
        return $this->getFileInfo()->getMimePart();
    }

    public function getType() {
        return $this->getFileInfo()->getType();
    }

    public function isEncrypted() {
        return $this->getFileInfo()->isEncrypted();
    }

    public function getMountPoint() {
        return $this->getFileInfo()->getMountPoint();
    }

    public function getOwner() {
        return $this->getFileInfo()->getOwner();
    }

    public function getChecksum() {
        return;
    }

    /**
     * @param int $type \OCP\Lock\ILockingProvider::LOCK_SHARED or \OCP\Lock\ILockingProvider::LOCK_EXCLUSIVE
     * @throws \OCP\Lock\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 \OCP\Lock\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 \OCP\Lock\LockedException
     */
    public function unlock($type) {
        $this->view->unlockFile($this->path, $type);
    }

    /**
     * @param string $targetPath
     * @throws \OCP\Files\NotPermittedException if copy not allowed or failed
     * @return \OC\Files\Node\Node
     */
    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->root->emit('\OC\Files', 'preCopy', [$this, $nonExisting]);
            $this->root->emit('\OC\Files', '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->root->emit('\OC\Files', 'postCopy', [$this, $targetNode]);
            $this->root->emit('\OC\Files', 'postWrite', [$targetNode]);
            return $targetNode;
        } else {
            throw new NotPermittedException('No permission to copy to path ' . $targetPath);
        }
    }

    /**
     * @param string $targetPath
     * @throws \OCP\Files\NotPermittedException if move not allowed or failed
     * @return \OC\Files\Node\Node
     */
    public function move($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->root->emit('\OC\Files', 'preRename', [$this, $nonExisting]);
            $this->root->emit('\OC\Files', 'preWrite', [$nonExisting]);
            if (!$this->view->rename($this->path, $targetPath)) {
                throw new NotPermittedException('Could not move ' . $this->path . ' to ' . $targetPath);
            }
            $targetNode = $this->root->get($targetPath);
            $this->root->emit('\OC\Files', 'postRename', [$this, $targetNode]);
            $this->root->emit('\OC\Files', 'postWrite', [$targetNode]);
            $this->path = $targetPath;
            return $targetNode;
        } else {
            throw new NotPermittedException('No permission to move to path ' . $targetPath);
        }
    }

    public function getView() {
        return $this->view;
    }
}