aboutsummaryrefslogtreecommitdiffstats
path: root/tests/lib/Files/FilesystemTest.php
diff options
context:
space:
mode:
Diffstat (limited to 'tests/lib/Files/FilesystemTest.php')
-rw-r--r--tests/lib/Files/FilesystemTest.php478
1 files changed, 478 insertions, 0 deletions
diff --git a/tests/lib/Files/FilesystemTest.php b/tests/lib/Files/FilesystemTest.php
new file mode 100644
index 00000000000..a819acb1620
--- /dev/null
+++ b/tests/lib/Files/FilesystemTest.php
@@ -0,0 +1,478 @@
+<?php
+
+/**
+ * SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace Test\Files;
+
+use OC\Files\Filesystem;
+use OC\Files\Mount\MountPoint;
+use OC\Files\Storage\Temporary;
+use OC\Files\View;
+use OC\User\NoUserException;
+use OCP\Files;
+use OCP\Files\Config\IMountProvider;
+use OCP\Files\Config\IMountProviderCollection;
+use OCP\Files\Mount\IMountPoint;
+use OCP\Files\Storage\IStorageFactory;
+use OCP\IConfig;
+use OCP\ITempManager;
+use OCP\IUser;
+use OCP\IUserManager;
+use OCP\IUserSession;
+use OCP\Server;
+
+class DummyMountProvider implements IMountProvider {
+ /**
+ * @param array $mounts
+ */
+ public function __construct(
+ private array $mounts,
+ ) {
+ }
+
+ /**
+ * Get the pre-registered mount points
+ *
+ * @param IUser $user
+ * @param IStorageFactory $loader
+ * @return IMountPoint[]
+ */
+ public function getMountsForUser(IUser $user, IStorageFactory $loader) {
+ return isset($this->mounts[$user->getUID()]) ? $this->mounts[$user->getUID()] : [];
+ }
+}
+
+/**
+ * Class FilesystemTest
+ *
+ * @group DB
+ *
+ * @package Test\Files
+ */
+class FilesystemTest extends \Test\TestCase {
+ public const TEST_FILESYSTEM_USER1 = 'test-filesystem-user1';
+ public const TEST_FILESYSTEM_USER2 = 'test-filesystem-user1';
+
+ /**
+ * @var array tmpDirs
+ */
+ private $tmpDirs = [];
+
+ /**
+ * @return array
+ */
+ private function getStorageData() {
+ $dir = Server::get(ITempManager::class)->getTemporaryFolder();
+ $this->tmpDirs[] = $dir;
+ return ['datadir' => $dir];
+ }
+
+ protected function setUp(): void {
+ parent::setUp();
+ $userBackend = new \Test\Util\User\Dummy();
+ $userBackend->createUser(self::TEST_FILESYSTEM_USER1, self::TEST_FILESYSTEM_USER1);
+ $userBackend->createUser(self::TEST_FILESYSTEM_USER2, self::TEST_FILESYSTEM_USER2);
+ Server::get(IUserManager::class)->registerBackend($userBackend);
+ $this->loginAsUser();
+ }
+
+ protected function tearDown(): void {
+ foreach ($this->tmpDirs as $dir) {
+ Files::rmdirr($dir);
+ }
+
+ $this->logout();
+ $this->invokePrivate('\OC\Files\Filesystem', 'normalizedPathCache', [null]);
+ parent::tearDown();
+ }
+
+ public function testMount(): void {
+ Filesystem::mount('\OC\Files\Storage\Local', self::getStorageData(), '/');
+ $this->assertEquals('/', Filesystem::getMountPoint('/'));
+ $this->assertEquals('/', Filesystem::getMountPoint('/some/folder'));
+ [, $internalPath] = Filesystem::resolvePath('/');
+ $this->assertEquals('', $internalPath);
+ [, $internalPath] = Filesystem::resolvePath('/some/folder');
+ $this->assertEquals('some/folder', $internalPath);
+
+ Filesystem::mount('\OC\Files\Storage\Local', self::getStorageData(), '/some');
+ $this->assertEquals('/', Filesystem::getMountPoint('/'));
+ $this->assertEquals('/some/', Filesystem::getMountPoint('/some/folder'));
+ $this->assertEquals('/some/', Filesystem::getMountPoint('/some/'));
+ $this->assertEquals('/some/', Filesystem::getMountPoint('/some'));
+ [, $internalPath] = Filesystem::resolvePath('/some/folder');
+ $this->assertEquals('folder', $internalPath);
+ }
+
+ public static function normalizePathData(): array {
+ return [
+ ['/', ''],
+ ['/', '/'],
+ ['/', '//'],
+ ['/', '/', false],
+ ['/', '//', false],
+
+ ['/path', '/path/'],
+ ['/path/', '/path/', false],
+ ['/path', 'path'],
+
+ ['/foo/bar', '/foo//bar/'],
+ ['/foo/bar/', '/foo//bar/', false],
+ ['/foo/bar', '/foo////bar'],
+ ['/foo/bar', '/foo/////bar'],
+ ['/foo/bar', '/foo/bar/.'],
+ ['/foo/bar', '/foo/bar/./'],
+ ['/foo/bar/', '/foo/bar/./', false],
+ ['/foo/bar', '/foo/bar/./.'],
+ ['/foo/bar', '/foo/bar/././'],
+ ['/foo/bar/', '/foo/bar/././', false],
+ ['/foo/bar', '/foo/./bar/'],
+ ['/foo/bar/', '/foo/./bar/', false],
+ ['/foo/.bar', '/foo/.bar/'],
+ ['/foo/.bar/', '/foo/.bar/', false],
+ ['/foo/.bar/tee', '/foo/.bar/tee'],
+ ['/foo/bar.', '/foo/bar./'],
+ ['/foo/bar./', '/foo/bar./', false],
+ ['/foo/bar./tee', '/foo/bar./tee'],
+ ['/foo/.bar.', '/foo/.bar./'],
+ ['/foo/.bar./', '/foo/.bar./', false],
+ ['/foo/.bar./tee', '/foo/.bar./tee'],
+
+ ['/foo/bar', '/.////././//./foo/.///././//./bar/././/./.'],
+ ['/foo/bar/', '/.////././//./foo/.///././//./bar/./././.', false],
+ ['/foo/bar', '/.////././//./foo/.///././//./bar/././/././'],
+ ['/foo/bar/', '/.////././//./foo/.///././//./bar/././/././', false],
+ ['/foo/.bar', '/.////././//./foo/./././/./.bar/././/././'],
+ ['/foo/.bar/', '/.////././//./foo/./././/./.bar/././/././', false],
+ ['/foo/.bar/tee./', '/.////././//./foo/./././/./.bar/tee././/././', false],
+ ['/foo/bar.', '/.////././//./foo/./././/./bar./././/././'],
+ ['/foo/bar./', '/.////././//./foo/./././/./bar./././/././', false],
+ ['/foo/bar./tee./', '/.////././//./foo/./././/./bar./tee././/././', false],
+ ['/foo/.bar.', '/.////././//./foo/./././/./.bar./././/././'],
+ ['/foo/.bar./', '/.////././//./foo/./././/./.bar./././././', false],
+ ['/foo/.bar./tee./', '/.////././//./foo/./././/./.bar./tee././././', false],
+
+ // Windows paths
+ ['/', ''],
+ ['/', '\\'],
+ ['/', '\\', false],
+ ['/', '\\\\'],
+ ['/', '\\\\', false],
+
+ ['/path', '\\path'],
+ ['/path', '\\path', false],
+ ['/path', '\\path\\'],
+ ['/path/', '\\path\\', false],
+
+ ['/foo/bar', '\\foo\\\\bar\\'],
+ ['/foo/bar/', '\\foo\\\\bar\\', false],
+ ['/foo/bar', '\\foo\\\\\\\\bar'],
+ ['/foo/bar', '\\foo\\\\\\\\\\bar'],
+ ['/foo/bar', '\\foo\\bar\\.'],
+ ['/foo/bar', '\\foo\\bar\\.\\'],
+ ['/foo/bar/', '\\foo\\bar\\.\\', false],
+ ['/foo/bar', '\\foo\\bar\\.\\.'],
+ ['/foo/bar', '\\foo\\bar\\.\\.\\'],
+ ['/foo/bar/', '\\foo\\bar\\.\\.\\', false],
+ ['/foo/bar', '\\foo\\.\\bar\\'],
+ ['/foo/bar/', '\\foo\\.\\bar\\', false],
+ ['/foo/.bar', '\\foo\\.bar\\'],
+ ['/foo/.bar/', '\\foo\\.bar\\', false],
+ ['/foo/.bar/tee', '\\foo\\.bar\\tee'],
+
+ // Absolute windows paths NOT marked as absolute
+ ['/C:', 'C:\\'],
+ ['/C:/', 'C:\\', false],
+ ['/C:/tests', 'C:\\tests'],
+ ['/C:/tests', 'C:\\tests', false],
+ ['/C:/tests', 'C:\\tests\\'],
+ ['/C:/tests/', 'C:\\tests\\', false],
+ ['/C:/tests/bar', 'C:\\tests\\.\\.\\bar'],
+ ['/C:/tests/bar/', 'C:\\tests\\.\\.\\bar\\.\\', false],
+
+ // normalize does not resolve '..' (by design)
+ ['/foo/..', '/foo/../'],
+ ['/foo/../bar', '/foo/../bar/.'],
+ ['/foo/..', '\\foo\\..\\'],
+ ['/foo/../bar', '\\foo\\..\\bar'],
+ ];
+ }
+
+ #[\PHPUnit\Framework\Attributes\DataProvider('normalizePathData')]
+ public function testNormalizePath($expected, $path, $stripTrailingSlash = true): void {
+ $this->assertEquals($expected, Filesystem::normalizePath($path, $stripTrailingSlash));
+ }
+
+ public static function normalizePathKeepUnicodeData(): array {
+ $nfdName = 'ümlaut';
+ $nfcName = 'ümlaut';
+ return [
+ ['/' . $nfcName, $nfcName, true],
+ ['/' . $nfcName, $nfcName, false],
+ ['/' . $nfdName, $nfdName, true],
+ ['/' . $nfcName, $nfdName, false],
+ ];
+ }
+
+ #[\PHPUnit\Framework\Attributes\DataProvider('normalizePathKeepUnicodeData')]
+ public function testNormalizePathKeepUnicode($expected, $path, $keepUnicode = false): void {
+ $this->assertEquals($expected, Filesystem::normalizePath($path, true, false, $keepUnicode));
+ }
+
+ public function testNormalizePathKeepUnicodeCache(): void {
+ $nfdName = 'ümlaut';
+ $nfcName = 'ümlaut';
+ // call in succession due to cache
+ $this->assertEquals('/' . $nfcName, Filesystem::normalizePath($nfdName, true, false, false));
+ $this->assertEquals('/' . $nfdName, Filesystem::normalizePath($nfdName, true, false, true));
+ }
+
+ public static function isValidPathData(): array {
+ return [
+ ['/', true],
+ ['/path', true],
+ ['/foo/bar', true],
+ ['/foo//bar/', true],
+ ['/foo////bar', true],
+ ['/foo//\///bar', true],
+ ['/foo/bar/.', true],
+ ['/foo/bar/./', true],
+ ['/foo/bar/./.', true],
+ ['/foo/bar/././', true],
+ ['/foo/bar/././..bar', true],
+ ['/foo/bar/././..bar/a', true],
+ ['/foo/bar/././..', false],
+ ['/foo/bar/././../', false],
+ ['/foo/bar/.././', false],
+ ['/foo/bar/../../', false],
+ ['/foo/bar/../..\\', false],
+ ['..', false],
+ ['../', false],
+ ['../foo/bar', false],
+ ['..\foo/bar', false],
+ ];
+ }
+
+ #[\PHPUnit\Framework\Attributes\DataProvider('isValidPathData')]
+ public function testIsValidPath($path, $expected): void {
+ $this->assertSame($expected, Filesystem::isValidPath($path));
+ }
+
+ public static function isFileBlacklistedData(): array {
+ return [
+ ['/etc/foo/bar/foo.txt', false],
+ ['\etc\foo/bar\foo.txt', false],
+ ['.htaccess', true],
+ ['.htaccess/', true],
+ ['.htaccess\\', true],
+ ['/etc/foo\bar/.htaccess\\', true],
+ ['/etc/foo\bar/.htaccess/', true],
+ ['/etc/foo\bar/.htaccess/foo', false],
+ ['//foo//bar/\.htaccess/', true],
+ ['\foo\bar\.HTAccess', true],
+ ];
+ }
+
+ #[\PHPUnit\Framework\Attributes\DataProvider('isFileBlacklistedData')]
+ public function testIsFileBlacklisted($path, $expected): void {
+ $this->assertSame($expected, Filesystem::isFileBlacklisted($path));
+ }
+
+ public function testNormalizePathUTF8(): void {
+ if (!class_exists('Patchwork\PHP\Shim\Normalizer')) {
+ $this->markTestSkipped('UTF8 normalizer Patchwork was not found');
+ }
+
+ $this->assertEquals("/foo/bar\xC3\xBC", Filesystem::normalizePath("/foo/baru\xCC\x88"));
+ $this->assertEquals("/foo/bar\xC3\xBC", Filesystem::normalizePath("\\foo\\baru\xCC\x88"));
+ }
+
+ public function testHooks(): void {
+ if (Filesystem::getView()) {
+ $user = \OC_User::getUser();
+ } else {
+ $user = self::TEST_FILESYSTEM_USER1;
+ $backend = new \Test\Util\User\Dummy();
+ Server::get(IUserManager::class)->registerBackend($backend);
+ $backend->createUser($user, $user);
+ $userObj = Server::get(IUserManager::class)->get($user);
+ Server::get(IUserSession::class)->setUser($userObj);
+ Filesystem::init($user, '/' . $user . '/files');
+ }
+ \OC_Hook::clear('OC_Filesystem');
+ \OC_Hook::connect('OC_Filesystem', 'post_write', $this, 'dummyHook');
+
+ Filesystem::mount('OC\Files\Storage\Temporary', [], '/');
+
+ $rootView = new View('');
+ $rootView->mkdir('/' . $user);
+ $rootView->mkdir('/' . $user . '/files');
+
+ // \OC\Files\Filesystem::file_put_contents('/foo', 'foo');
+ Filesystem::mkdir('/bar');
+ // \OC\Files\Filesystem::file_put_contents('/bar//foo', 'foo');
+
+ $tmpFile = Server::get(ITempManager::class)->getTemporaryFile();
+ file_put_contents($tmpFile, 'foo');
+ $fh = fopen($tmpFile, 'r');
+ // \OC\Files\Filesystem::file_put_contents('/bar//foo', $fh);
+ }
+
+ /**
+ * Tests that an exception is thrown when passed user does not exist.
+ *
+ */
+ public function testLocalMountWhenUserDoesNotExist(): void {
+ $this->expectException(NoUserException::class);
+
+ $userId = $this->getUniqueID('user_');
+
+ Filesystem::initMountPoints($userId);
+ }
+
+
+ public function testNullUserThrows(): void {
+ $this->expectException(NoUserException::class);
+
+ Filesystem::initMountPoints(null);
+ }
+
+ public function testNullUserThrowsTwice(): void {
+ $thrown = 0;
+ try {
+ Filesystem::initMountPoints(null);
+ } catch (NoUserException $e) {
+ $thrown++;
+ }
+ try {
+ Filesystem::initMountPoints(null);
+ } catch (NoUserException $e) {
+ $thrown++;
+ }
+ $this->assertEquals(2, $thrown);
+ }
+
+ /**
+ * Tests that an exception is thrown when passed user does not exist.
+ */
+ public function testLocalMountWhenUserDoesNotExistTwice(): void {
+ $thrown = 0;
+ $userId = $this->getUniqueID('user_');
+
+ try {
+ Filesystem::initMountPoints($userId);
+ } catch (NoUserException $e) {
+ $thrown++;
+ }
+
+ try {
+ Filesystem::initMountPoints($userId);
+ } catch (NoUserException $e) {
+ $thrown++;
+ }
+
+ $this->assertEquals(2, $thrown);
+ }
+
+ /**
+ * Tests that the home storage is used for the user's mount point
+ */
+ public function testHomeMount(): void {
+ $userId = $this->getUniqueID('user_');
+
+ Server::get(IUserManager::class)->createUser($userId, $userId);
+
+ Filesystem::initMountPoints($userId);
+
+ $homeMount = Filesystem::getStorage('/' . $userId . '/');
+
+ $this->assertTrue($homeMount->instanceOfStorage('\OCP\Files\IHomeStorage'));
+ if ($homeMount->instanceOfStorage('\OC\Files\ObjectStore\HomeObjectStoreStorage')) {
+ $this->assertEquals('object::user:' . $userId, $homeMount->getId());
+ } elseif ($homeMount->instanceOfStorage('\OC\Files\Storage\Home')) {
+ $this->assertEquals('home::' . $userId, $homeMount->getId());
+ }
+
+ $user = Server::get(IUserManager::class)->get($userId);
+ if ($user !== null) {
+ $user->delete();
+ }
+ }
+
+ public function dummyHook($arguments) {
+ $path = $arguments['path'];
+ $this->assertEquals($path, Filesystem::normalizePath($path)); //the path passed to the hook should already be normalized
+ }
+
+ /**
+ * Test that the default cache dir is part of the user's home
+ */
+ public function testMountDefaultCacheDir(): void {
+ $userId = $this->getUniqueID('user_');
+ $config = Server::get(IConfig::class);
+ $oldCachePath = $config->getSystemValueString('cache_path', '');
+ // no cache path configured
+ $config->setSystemValue('cache_path', '');
+
+ Server::get(IUserManager::class)->createUser($userId, $userId);
+ Filesystem::initMountPoints($userId);
+
+ $this->assertEquals(
+ '/' . $userId . '/',
+ Filesystem::getMountPoint('/' . $userId . '/cache')
+ );
+ [$storage, $internalPath] = Filesystem::resolvePath('/' . $userId . '/cache');
+ $this->assertTrue($storage->instanceOfStorage('\OCP\Files\IHomeStorage'));
+ $this->assertEquals('cache', $internalPath);
+ $user = Server::get(IUserManager::class)->get($userId);
+ if ($user !== null) {
+ $user->delete();
+ }
+
+ $config->setSystemValue('cache_path', $oldCachePath);
+ }
+
+ /**
+ * Test that an external cache is mounted into
+ * the user's home
+ */
+ public function testMountExternalCacheDir(): void {
+ $userId = $this->getUniqueID('user_');
+
+ $config = Server::get(IConfig::class);
+ $oldCachePath = $config->getSystemValueString('cache_path', '');
+ // set cache path to temp dir
+ $cachePath = Server::get(ITempManager::class)->getTemporaryFolder() . '/extcache';
+ $config->setSystemValue('cache_path', $cachePath);
+
+ Server::get(IUserManager::class)->createUser($userId, $userId);
+ Filesystem::initMountPoints($userId);
+
+ $this->assertEquals(
+ '/' . $userId . '/cache/',
+ Filesystem::getMountPoint('/' . $userId . '/cache')
+ );
+ [$storage, $internalPath] = Filesystem::resolvePath('/' . $userId . '/cache');
+ $this->assertTrue($storage->instanceOfStorage('\OC\Files\Storage\Local'));
+ $this->assertEquals('', $internalPath);
+ $user = Server::get(IUserManager::class)->get($userId);
+ if ($user !== null) {
+ $user->delete();
+ }
+
+ $config->setSystemValue('cache_path', $oldCachePath);
+ }
+
+ public function testRegisterMountProviderAfterSetup(): void {
+ Filesystem::initMountPoints(self::TEST_FILESYSTEM_USER2);
+ $this->assertEquals('/', Filesystem::getMountPoint('/foo/bar'));
+ $mount = new MountPoint(new Temporary([]), '/foo/bar');
+ $mountProvider = new DummyMountProvider([self::TEST_FILESYSTEM_USER2 => [$mount]]);
+ Server::get(IMountProviderCollection::class)->registerProvider($mountProvider);
+ $this->assertEquals('/foo/bar/', Filesystem::getMountPoint('/foo/bar'));
+ }
+}