// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE namespace QTest { template<> char *toString(const Utils::FilePath &filePath) { return qstrdup(filePath.toUrlishString().toLocal8Bit().constData()); } } // namespace QTest QT_END_NAMESPACE namespace Utils { void ignoreSoftAssert() { QTest::ignoreMessage(QtDebugMsg, QRegularExpression("SOFT ASSERT.*")); } class tst_filepath : public QObject { Q_OBJECT private slots: void initTestCase(); void isEmpty_data(); void isEmpty(); void parentDir_data(); void parentDir(); void isChildOf_data(); void isChildOf(); void fileName_data(); void fileName(); void relativePathFromDir_specials(); void relativePathFromDir_data(); void relativePathFromDir(); void absolute_data(); void absolute(); void fromToString_data(); void fromToString(); void fromString_data(); void fromString(); void fromUserInput_data(); void fromUserInput(); void toString_data(); void toString(); void toFSPathString_data(); void toFSPathString(); void comparison_data(); void comparison(); void linkFromString_data(); void linkFromString(); void pathAppended_data(); void pathAppended(); void resolvePath_filepath_data(); void resolvePath_filepath(); void resolvePath_string_data(); void resolvePath_string(); void relativeChildPath_data(); void relativeChildPath(); void rootLength_data(); void rootLength(); void schemeAndHostLength_data(); void schemeAndHostLength(); void asyncLocalCopy(); void startsWithDriveLetter(); void startsWithDriveLetter_data(); void withNewMappedPath_data(); void withNewMappedPath(); void stringAppended(); void stringAppended_data(); void url(); void url_data(); void cleanPath_data(); void cleanPath(); void isSameFile_data(); void isSameFile(); void hostSpecialChars_data(); void hostSpecialChars(); void tmp(); void tmp_data(); void searchInWithFilter(); void sort(); void sort_data(); void isRootPath(); void isRootPath_data(); void lessThan(); void lessThan_data(); void asQMapKey(); void makeTemporaryFile(); void dontBreakPathOnWierdWindowsPaths(); void isRelativePath(); void isRelativePath_data(); void pathComponents(); void pathComponents_data(); void symLinks(); void resolveSymLinks(); void ensureWritableDirectory(); void ensureWritableDirectoryPermissions(); void searchHereAndInParents(); void parents(); void emptyParents(); void parentsWithDevice(); void parentsWithDrive(); void parentsWithUncPath(); void parentsWithLastPath(); void exists(); void isNewerThan(); void watch(); void coroTest(); private: QTemporaryDir tempDir; QString rootPath; QString exeExt; }; static bool touch(const QDir &dir, const QString &filename, bool fill, bool executable = false) { QFile file(dir.absoluteFilePath(filename)); if (!file.open(QIODevice::WriteOnly)) return false; if (executable) { if (!file.setPermissions(file.permissions() | QFileDevice::ExeUser)) return false; } if (fill) { QRandomGenerator *random = QRandomGenerator::global(); for (int i = 0; i < 10; ++i) file.write(QString::number(random->generate(), 16).toUtf8()); } file.close(); return true; } void tst_filepath::initTestCase() { // initialize test for tst_filepath::relativePath*() QVERIFY(tempDir.isValid()); rootPath = QFileInfo(tempDir.path()).canonicalFilePath(); QDir dir(rootPath); dir.mkpath("a/b/c/d"); dir.mkpath("a/x/y/z"); dir.mkpath("a/b/x/y/z"); dir.mkpath("x/y/z"); QVERIFY(touch(dir, "a/b/c/d/file1.txt", false)); QVERIFY(touch(dir, "a/x/y/z/file2.txt", false)); QVERIFY(touch(dir, "a/file3.txt", false)); QVERIFY(touch(dir, "x/y/file4.txt", false)); // initialize test for tst_filepath::asyncLocalCopy() QVERIFY(touch(dir, "x/y/fileToCopy.txt", true)); // initialize test for tst_filepath::searchIn() #ifdef Q_OS_WIN exeExt = ".exe"; #endif dir.mkpath("s/1"); dir.mkpath("s/2"); QVERIFY(touch(dir, "s/1/testexe" + exeExt, false, true)); QVERIFY(touch(dir, "s/2/testexe" + exeExt, false, true)); } void tst_filepath::searchInWithFilter() { const FilePaths dirs = {FilePath::fromUserInput(rootPath) / "s" / "1", FilePath::fromUserInput(rootPath) / "s" / "2"}; FilePath exe = FilePath::fromUserInput("testexe" + exeExt) .searchInDirectories(dirs, [](const FilePath &path) { return path.path().contains("/2/"); }); QVERIFY(!exe.path().endsWith("/1/testexe" + exeExt) && exe.path().endsWith("/2/testexe" + exeExt)); FilePath exe2 = FilePath::fromUserInput("testexe" + exeExt) .searchInDirectories(dirs, [](const FilePath &path) { return path.path().contains("/1/"); }); QVERIFY(!exe2.path().endsWith("/2/testexe" + exeExt) && exe2.path().endsWith("/1/testexe" + exeExt)); } void tst_filepath::isEmpty_data() { QTest::addColumn("path"); QTest::addColumn("result"); QTest::newRow("empty path") << "" << true; QTest::newRow("root only") << "/" << false; QTest::newRow("//") << "//" << false; QTest::newRow("scheme://host") << "scheme://host" << true; // Intentional (for now?) QTest::newRow("scheme://host/") << "scheme://host/" << false; QTest::newRow("scheme://host/a") << "scheme://host/a" << false; QTest::newRow("scheme://host/.") << "scheme://host/." << false; } void tst_filepath::coroTest() { Result<> res = [this]() -> Result<> { const FilePath dir = FilePath::fromUserInput(tempDir.path()) / "coro_test_dir"; co_await dir.ensureWritableDir(); const FilePath file = dir / "coro_test_file.txt"; const QByteArray data = "Hello, coroutine!"; co_await file.writeFileContents(data); QByteArray readData = co_await file.fileContents(); if (readData != data) // Unlikely to happen co_return ResultError("Data read does not match data written"); // We want it to fail to have an example of error handling const FilePath nonExistentFile = dir / "non_existent_file.txt"; readData = co_await nonExistentFile.fileContents(); qDebug() << "This line should not be reached, as the previous co_await should fail"; co_return ResultOk; }(); QEXPECT_FAIL("", "Expected failure reading non-existent file", Continue); QVERIFY_RESULT(res); } void tst_filepath::isEmpty() { QFETCH(QString, path); QFETCH(bool, result); FilePath filePath = FilePath::fromString(path); QCOMPARE(filePath.isEmpty(), result); } void tst_filepath::parentDir_data() { QTest::addColumn("path"); QTest::addColumn("parentPath"); QTest::addColumn("expectFailMessage"); QTest::newRow("empty path") << "" << "" << ""; QTest::newRow("//") << "//" << "//" << ""; QTest::newRow("/tmp/dir") << "/tmp/dir" << "/tmp" << ""; QTest::newRow("relative/path") << "relative/path" << "relative" << ""; QTest::newRow("relativepath") << "relativepath" << "." << ""; // Windows stuff: QTest::newRow("C:/data") << "C:/data" << "C:/" << ""; QTest::newRow("//./com1") << "//./com1" << "//./" << ""; QTest::newRow("//?/path") << "//?/path" << "/" << "Qt 4 cannot handle this path."; QTest::newRow("/Global?\?/UNC/host") << "/Global?\?/UNC/host" << "/Global?\?/UNC/host" << "Qt 4 cannot handle this path."; QTest::newRow("//server/directory/file") << "//server/directory/file" << "//server/directory" << ""; QTest::newRow("//server/directory") << "//server/directory" << "//server/" << ""; QTest::newRow("//server") << "//server" << "//server" << ""; QTest::newRow("qrc") << ":/foo/bar.txt" << ":/foo" << ""; QTest::newRow("root only") << "/" << "/" << ""; QTest::newRow("C:/") << "C:/" << "C:/" << ""; QTest::newRow("D:/") << "D:/" << "D:/" << ""; } void tst_filepath::parentDir() { QFETCH(QString, path); QFETCH(QString, parentPath); QFETCH(QString, expectFailMessage); FilePath result = FilePath::fromUserInput(path).parentDir(); if (!expectFailMessage.isEmpty()) QEXPECT_FAIL("", expectFailMessage.toUtf8().constData(), Continue); QCOMPARE(result.toUrlishString(), parentPath); } void tst_filepath::isChildOf_data() { QTest::addColumn("path"); QTest::addColumn("childPath"); QTest::addColumn("result"); QTest::newRow("empty path") << "" << "/tmp" << false; QTest::newRow("root only") << "/" << "/tmp" << true; QTest::newRow("/tmp/dir") << "/tmp" << "/tmp/dir" << true; QTest::newRow("relative/path") << "relative" << "relative/path" << true; QTest::newRow("/tmpdir") << "/tmp" << "/tmpdir" << false; QTest::newRow("same-0") << "/tmp/dir" << "/tmp/dir" << false; QTest::newRow("same-1") << "/tmp/dir/" << "/tmp/dir" << false; QTest::newRow("same-2") << "/tmp/dir" << "/tmp/dir/" << false; // Windows stuff: QTest::newRow("C:/data") << "C:/" << "C:/data" << true; QTest::newRow("C:/") << "" << "C:/" << false; QTest::newRow("com-port") << "//./" << "//./com1" << true; QTest::newRow("extended-length-path") << "\\\\?\\C:\\" << "\\\\?\\C:\\path" << true; QTest::newRow("/Global?\?/UNC/host") << "/Global?\?/UNC/host" << "/Global?\?/UNC/host/file" << true; QTest::newRow("//server/directory/file") << "//server/directory" << "//server/directory/file" << true; QTest::newRow("//server/directory") << "//server" << "//server/directory" << true; QTest::newRow("qrc") << ":/foo/bar" << ":/foo/bar/blah" << true; } void tst_filepath::isChildOf() { QFETCH(QString, path); QFETCH(QString, childPath); QFETCH(bool, result); const FilePath child = FilePath::fromUserInput(childPath); const FilePath parent = FilePath::fromUserInput(path); QCOMPARE(child.isChildOf(parent), result); } void tst_filepath::fileName_data() { QTest::addColumn("path"); QTest::addColumn("components"); QTest::addColumn("result"); QTest::newRow("empty 1") << "" << 0 << ""; QTest::newRow("empty 2") << "" << 1 << ""; QTest::newRow("basic") << "/foo/bar/baz" << 0 << "baz"; QTest::newRow("2 parts") << "/foo/bar/baz" << 1 << "bar/baz"; QTest::newRow("root no depth") << "/foo" << 0 << "foo"; QTest::newRow("root full") << "/foo" << 1 << "/foo"; QTest::newRow("root included") << "/foo/bar/baz" << 2 << "/foo/bar/baz"; QTest::newRow("too many parts") << "/foo/bar/baz" << 5 << "/foo/bar/baz"; QTest::newRow("windows root") << "C:/foo/bar/baz" << 2 << "C:/foo/bar/baz"; QTest::newRow("smb share") << "//server/share/file" << 2 << "//server/share/file"; QTest::newRow("no slashes") << "foobar" << 0 << "foobar"; QTest::newRow("no slashes with depth") << "foobar" << 1 << "foobar"; QTest::newRow("multiple slashes 1") << "/foo/bar////baz" << 0 << "baz"; QTest::newRow("multiple slashes 2") << "/foo/bar////baz" << 1 << "bar////baz"; QTest::newRow("multiple slashes 3") << "/foo////bar/baz" << 2 << "/foo////bar/baz"; QTest::newRow("single char 1") << "/a/b/c" << 0 << "c"; QTest::newRow("single char 2") << "/a/b/c" << 1 << "b/c"; QTest::newRow("single char 3") << "/a/b/c" << 2 << "/a/b/c"; QTest::newRow("slash at end 1") << "/a/b/" << 0 << ""; QTest::newRow("slash at end 2") << "/a/b/" << 1 << "b/"; QTest::newRow("slashes at end 1") << "/a/b//" << 0 << ""; QTest::newRow("slashes at end 2") << "/a/b//" << 1 << "b//"; QTest::newRow("root only 1") << "/" << 0 << ""; QTest::newRow("root only 2") << "/" << 1 << "/"; QTest::newRow("qrc 0") << ":/foo/bar" << 0 << "bar"; QTest::newRow("qrc with root") << ":/foo/bar" << 1 << ":/foo/bar"; } void tst_filepath::fileName() { QFETCH(QString, path); QFETCH(int, components); QFETCH(QString, result); QCOMPARE(FilePath::fromString(path).fileNameWithPathComponents(components), result); } void tst_filepath::relativePathFromDir_specials() { QString path = FilePath("").relativePathFromDir(""); QCOMPARE(path, ""); } void tst_filepath::relativePathFromDir_data() { QTest::addColumn("current"); QTest::addColumn("anchor"); QTest::addColumn("result"); QTest::newRow("samedir") << "a/b/c/d" << "a/b/c/d" << "."; QTest::newRow("samedir_but_file") << "a/b/c/d/file1.txt" << "a/b/c/d" << "file1.txt"; QTest::newRow("dir2dir_1") << "a/b/c/d" << "a/x/y/z" << "../../../b/c/d"; QTest::newRow("dir2dir_2") << "a/b" << "a/b/c" << ".."; QTest::newRow("file2dir_1") << "a/b/c/d/file1.txt" << "x/y" << "../../a/b/c/d/file1.txt"; QTest::newRow("abs_samedir_but_file") << "/a/b/c/d/file1.txt" << "/a/b/c/d" << "file1.txt"; QTest::newRow("abs_dir2dir_1") << "/a/b/c/d" << "/a/x/y/z" << "../../../b/c/d"; QTest::newRow("abs_dir2dir_2") << "/a/b" << "/a/b/c" << ".."; QTest::newRow("abs_file2dir_1") << "/a/b/c/d/file1.txt" << "/x/y" << "../../a/b/c/d/file1.txt"; QTest::newRow("remote_samedir_but_file") << "ssh://1.2.3.4/a/b/c/d/file1.txt" << "ssh://1.2.3.4/a/b/c/d" << "file1.txt"; QTest::newRow("remote_dir2dir_1") << "ssh://1.2.3.4/a/b/c/d" << "ssh://1.2.3.4/a/x/y/z" << "../../../b/c/d"; QTest::newRow("remote_dir2dir_2") << "ssh://1.2.3.4/a/b" << "ssh://1.2.3.4/a/b/c" << ".."; QTest::newRow("remote_file2dir_1") << "ssh://1.2.3.4/a/b/c/d/file1.txt" << "ssh://1.2.3.4/x/y" << "../../a/b/c/d/file1.txt"; /// QTest::newRow("empty") << "" << "" << ""; QTest::newRow("leftempty") << "" << "/" << ""; QTest::newRow("rightempty") << "/" << "" << ""; QTest::newRow("root") << "/" << "/" << "."; QTest::newRow("simple1") << "/a" << "/" << "a"; QTest::newRow("simple2") << "/" << "/a" << ".."; QTest::newRow("simple3") << "/a" << "/a" << "."; QTest::newRow("extraslash1") << "/a/b/c" << "/a/b/c" << "."; QTest::newRow("extraslash2") << "/a/b/c" << "/a/b/c/" << "."; QTest::newRow("extraslash3") << "/a/b/c/" << "/a/b/c" << "."; QTest::newRow("normal1") << "/a/b/c" << "/a/x" << "../b/c"; QTest::newRow("normal2") << "/a/b/c" << "/a/x/y" << "../../b/c"; QTest::newRow("normal3") << "/a/b/c" << "/x/y" << "../../a/b/c"; if (HostOsInfo::isWindowsHost()) { QTest::newRow("different drive letter case") << "C:/myproject/main.cpp" << "c:/myproject" << "main.cpp"; } } void tst_filepath::relativePathFromDir() { QFETCH(QString, current); QFETCH(QString, anchor); QFETCH(QString, result); QString actualPath = FilePath::fromString(current) .relativePathFromDir(FilePath::fromString(anchor)); QCOMPARE(actualPath, result); } void tst_filepath::rootLength_data() { QTest::addColumn("path"); QTest::addColumn("result"); QTest::newRow("empty") << "" << 0; QTest::newRow("slash") << "/" << 1; QTest::newRow("slash-rest") << "/abc" << 1; QTest::newRow("rest") << "abc" << 0; QTest::newRow("drive-slash") << "x:/" << 3; QTest::newRow("drive-rest") << "x:abc" << 0; QTest::newRow("drive-slash-rest") << "x:/abc" << 3; QTest::newRow("unc-root") << "//" << 2; QTest::newRow("unc-localhost-unfinished") << "//localhost" << 11; QTest::newRow("unc-localhost") << "//localhost/" << 12; QTest::newRow("unc-localhost-rest") << "//localhost/abs" << 12; QTest::newRow("unc-localhost-drive") << "//localhost/c$" << 12; QTest::newRow("unc-localhost-drive-slash") << "//localhost//c$/" << 12; QTest::newRow("unc-localhost-drive-slash-rest") << "//localhost//c$/x" << 12; QTest::newRow("windows-1") << "C:" << 2; QTest::newRow("windows-2") << "C:/" << 3; QTest::newRow("windows-3") << "C:/foor" << 3; } void tst_filepath::rootLength() { QFETCH(QString, path); QFETCH(int, result); int actual = FilePath::rootLength(path); QCOMPARE(actual, result); } void tst_filepath::schemeAndHostLength_data() { QTest::addColumn("path"); QTest::addColumn("result"); QTest::newRow("empty") << "" << 0; QTest::newRow("drive-slash-rest") << "x:/abc" << 0; QTest::newRow("rest") << "abc" << 0; QTest::newRow("slash-rest") << "/abc" << 0; QTest::newRow("dev-empty") << "dev://" << 6; QTest::newRow("dev-localhost-unfinished") << "dev://localhost" << 15; QTest::newRow("dev-localhost") << "dev://localhost/" << 16; QTest::newRow("dev-localhost-rest") << "dev://localhost/abs" << 16; QTest::newRow("dev-localhost-drive") << "dev://localhost/c$" << 16; QTest::newRow("dev-localhost-drive-slash") << "dev://localhost//c$/" << 16; QTest::newRow("dev-localhost-drive-slash-rest") << "dev://localhost//c$/x" << 16; } void tst_filepath::schemeAndHostLength() { QFETCH(QString, path); QFETCH(int, result); int actual = FilePath::schemeAndHostLength(path); QCOMPARE(actual, result); } void tst_filepath::absolute_data() { QTest::addColumn("path"); QTest::addColumn("absoluteFilePath"); QTest::addColumn("absolutePath"); QTest::newRow("absolute1") << FilePath::fromString("/") << FilePath::fromString("/") << FilePath::fromString("/"); QTest::newRow("absolute2") << FilePath::fromString("C:/a/b") << FilePath::fromString("C:/a/b") << FilePath::fromString("C:/a"); QTest::newRow("absolute3") << FilePath::fromString("/a/b") << FilePath::fromString("/a/b") << FilePath::fromString("/a"); QTest::newRow("absolute4") << FilePath::fromString("/a/b/..") << FilePath::fromString("/a") << FilePath::fromString("/"); QTest::newRow("absolute5") << FilePath::fromString("/a/b/c/../d") << FilePath::fromString("/a/b/d") << FilePath::fromString("/a/b"); QTest::newRow("absolute6") << FilePath::fromString("/a/../b/c/d") << FilePath::fromString("/b/c/d") << FilePath::fromString("/b/c"); QTest::newRow("default-constructed") << FilePath() << FilePath() << FilePath(); QTest::newRow("relative") << FilePath::fromString("a/b") << FilePath::fromString(QDir::currentPath() + "/a/b") << FilePath::fromString(QDir::currentPath() + "/a"); QTest::newRow("qrc") << FilePath::fromString(":/foo/bar.txt") << FilePath::fromString(":/foo/bar.txt") << FilePath::fromString(":/foo"); } void tst_filepath::absolute() { QFETCH(FilePath, path); QFETCH(FilePath, absoluteFilePath); QFETCH(FilePath, absolutePath); QCOMPARE(path.absoluteFilePath(), absoluteFilePath); QCOMPARE(path.absolutePath(), absolutePath); } void tst_filepath::toString_data() { QTest::addColumn("scheme"); QTest::addColumn("host"); QTest::addColumn("path"); QTest::addColumn("result"); QTest::addColumn("userResult"); QTest::newRow("empty") << "" << "" << "" << "" << ""; QTest::newRow("scheme") << "http" << "" << "" << "http://" << "http://"; QTest::newRow("scheme-and-host") << "http" << "127.0.0.1" << "" << "http://127.0.0.1" << "http://127.0.0.1"; QTest::newRow("root") << "http" << "127.0.0.1" << "/" << "http://127.0.0.1/" << "http://127.0.0.1/"; QTest::newRow("root-folder") << "" << "" << "/" << "/" << "/"; QTest::newRow("qtc-dev-root-folder-linux") << "" << "" << "/__qtc_devices__" << "/__qtc_devices__" << "/__qtc_devices__"; QTest::newRow("qtc-dev-root-folder-win") << "" << "" << "c:/__qtc_devices__" << "c:/__qtc_devices__" << "c:/__qtc_devices__"; QTest::newRow("qtc-dev-type-root-folder-linux") << "" << "" << "/__qtc_devices__/docker" << "/__qtc_devices__/docker" << "/__qtc_devices__/docker"; QTest::newRow("qtc-dev-type-root-folder-win") << "" << "" << "c:/__qtc_devices__/docker" << "c:/__qtc_devices__/docker" << "c:/__qtc_devices__/docker"; QTest::newRow("qtc-root-folder") << "docker" << "alpine.latest" << "/" << "docker://alpine.latest/" << "docker://alpine.latest/"; QTest::newRow("qtc-root-folder-rel") << "docker" << "alpine.latest" << "" << "docker://alpine.latest" << "docker://alpine.latest"; } void tst_filepath::toString() { QFETCH(QString, scheme); QFETCH(QString, host); QFETCH(QString, path); QFETCH(QString, result); QFETCH(QString, userResult); FilePath filePath = FilePath::fromParts(scheme, host, path); QCOMPARE(filePath.toUrlishString(), result); QString cleanedOutput = filePath.isLocal() ? QDir::cleanPath(filePath.toUserOutput()) : filePath.toUserOutput(); QCOMPARE(cleanedOutput, userResult); } void tst_filepath::toFSPathString_data() { QTest::addColumn("scheme"); QTest::addColumn("host"); QTest::addColumn("path"); QTest::addColumn("result"); QTest::addColumn("userResult"); QTest::newRow("empty") << "" << "" << "" << "" << ""; QTest::newRow("scheme") << "http" << "" << "" << QDir::rootPath() + "__qtc_devices__/http/" << "http://"; QTest::newRow("scheme-and-host") << "http" << "127.0.0.1" << "" << QDir::rootPath() + "__qtc_devices__/http/127.0.0.1" << "http://127.0.0.1"; QTest::newRow("root") << "http" << "127.0.0.1" << "/" << QDir::rootPath() + "__qtc_devices__/http/127.0.0.1/" << "http://127.0.0.1/"; QTest::newRow("root-folder") << "" << "" << "/" << "/" << "/"; QTest::newRow("qtc-dev-root-folder") << "" << "" << QDir::rootPath() + "__qtc_devices__" << QDir::rootPath() + "__qtc_devices__" << QDir::rootPath() + "__qtc_devices__"; QTest::newRow("qtc-dev-type-root-folder") << "" << "" << QDir::rootPath() + "__qtc_devices__/docker" << QDir::rootPath() + "__qtc_devices__/docker" << QDir::rootPath() + "__qtc_devices__/docker"; QTest::newRow("qtc-root-folder") << "docker" << "alpine.latest" << "/" << QDir::rootPath() + "__qtc_devices__/docker/alpine.latest/" << "docker://alpine.latest/"; QTest::newRow("qtc-root-folder-rel") << "docker" << "alpine.latest" << "" << QDir::rootPath() + "__qtc_devices__/docker/alpine.latest" << "docker://alpine.latest"; } void tst_filepath::toFSPathString() { QFETCH(QString, scheme); QFETCH(QString, host); QFETCH(QString, path); QFETCH(QString, result); QFETCH(QString, userResult); FilePath filePath = FilePath::fromParts(scheme, host, path); QCOMPARE(filePath.toFSPathString(), result); QString cleanedOutput = filePath.isLocal() ? QDir::cleanPath(filePath.toUserOutput()) : filePath.toUserOutput(); QCOMPARE(cleanedOutput, userResult); } enum ExpectedPass { PassEverywhere = 0, FailOnWindows = 1, FailOnLinux = 2, FailEverywhere = 3 }; class FromStringData { public: FromStringData(const QString &input, const QString &scheme, const QString &host, const QString &path, ExpectedPass expectedPass = PassEverywhere) : input(input) , scheme(scheme) , host(host) , path(path) , expectedPass(expectedPass) {} QString input; QString scheme; QString host; QString path; ExpectedPass expectedPass = PassEverywhere; }; } // Utils Q_DECLARE_METATYPE(Utils::FromStringData); namespace Utils { void tst_filepath::fromString_data() { using D = FromStringData; QTest::addColumn("data"); QTest::newRow("empty") << D("", "", "", ""); QTest::newRow("single-colon") << D(":", "", "", ":"); QTest::newRow("single-slash") << D("/", "", "", "/"); QTest::newRow("single-char") << D("a", "", "", "a"); QTest::newRow("relative") << D("./rel", "", "", "./rel"); QTest::newRow("qrc") << D(":/test.txt", "", "", ":/test.txt"); QTest::newRow("qrc-no-slash") << D(":test.txt", "", "", ":test.txt"); QTest::newRow("unc-incomplete") << D("//", "", "", "//"); QTest::newRow("unc-incomplete-only-server") << D("//server", "", "", "//server"); QTest::newRow("unc-incomplete-only-server-2") << D("//server/", "", "", "//server/"); QTest::newRow("unc-server-and-share") << D("//server/share", "", "", "//server/share"); QTest::newRow("unc-server-and-share-2") << D("//server/share/", "", "", "//server/share/"); QTest::newRow("unc-full") << D("//server/share/test.txt", "", "", "//server/share/test.txt"); QTest::newRow("unix-root") << D("/", "", "", "/"); QTest::newRow("unix-folder") << D("/tmp", "", "", "/tmp"); QTest::newRow("unix-folder-with-trailing-slash") << D("/tmp/", "", "", "/tmp/"); QTest::newRow("windows-root") << D("c:", "", "", "c:"); QTest::newRow("windows-folder") << D("c:/Windows", "", "", "c:/Windows"); QTest::newRow("windows-folder-with-trailing-slash") << D("c:/Windows/", "", "", "c:/Windows/"); QTest::newRow("windows-folder-slash") << D("C:/Windows", "", "", "C:/Windows"); QTest::newRow("docker-root-url") << D("docker://1234/", "docker", "1234", "/"); QTest::newRow("docker-root-url-special-linux") << D("/__qtc_devices__/docker/1234/", "docker", "1234", "/"); QTest::newRow("docker-root-url-special-win") << D("c:/__qtc_devices__/docker/1234/", "docker", "1234", "/"); QTest::newRow("docker-relative-path") << D("docker://1234/./rel", "docker", "1234", "rel"); QTest::newRow("qtc-dev-linux") << D("/__qtc_devices__", "", "", "/__qtc_devices__"); QTest::newRow("qtc-dev-win") << D("c:/__qtc_devices__", "", "", "c:/__qtc_devices__"); QTest::newRow("qtc-dev-type-linux") << D("/__qtc_devices__/docker", "", "", "/__qtc_devices__/docker"); QTest::newRow("qtc-dev-type-win") << D("c:/__qtc_devices__/docker", "", "", "c:/__qtc_devices__/docker"); QTest::newRow("qtc-dev-type-dev-linux") << D("/__qtc_devices__/docker/1234", "docker", "1234", "/"); QTest::newRow("qtc-dev-type-dev-win") << D("c:/__qtc_devices__/docker/1234", "docker", "1234", "/"); // "Remote Windows" is currently truly not supported. QTest::newRow("cross-os-linux") << D("/__qtc_devices__/docker/1234/c:/test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("cross-os-win") << D("c:/__qtc_devices__/docker/1234/c:/test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("cross-os-unclean-linux") << D("/__qtc_devices__/docker/1234/c:\\test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("cross-os-unclean-win") << D("c:/__qtc_devices__/docker/1234/c:\\test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("unc-full-in-docker-linux") << D("/__qtc_devices__/docker/1234//server/share/test.txt", "docker", "1234", "//server/share/test.txt"); QTest::newRow("unc-full-in-docker-win") << D("c:/__qtc_devices__/docker/1234//server/share/test.txt", "docker", "1234", "//server/share/test.txt"); QTest::newRow("unc-dos-1") << D("//?/c:", "", "", "//?/c:"); QTest::newRow("unc-dos-com") << D("//./com1", "", "", "//./com1"); } void tst_filepath::fromString() { QFETCH(FromStringData, data); FilePath filePath = FilePath::fromString(data.input); bool expectFail = ((data.expectedPass & FailOnLinux) && !HostOsInfo::isWindowsHost()) || ((data.expectedPass & FailOnWindows) && HostOsInfo::isWindowsHost()); if (expectFail) { QString actual = filePath.scheme() + '|' + filePath.host() + '|' + filePath.path(); QString expected = data.scheme + '|' + data.host + '|' + data.path; QEXPECT_FAIL("", "", Continue); QCOMPARE(actual, expected); return; } QCOMPARE(filePath.scheme(), data.scheme); QCOMPARE(filePath.host(), data.host); QCOMPARE(filePath.path(), data.path); } void tst_filepath::fromUserInput_data() { using D = FromStringData; QTest::addColumn("data"); QTest::newRow("empty") << D("", "", "", ""); QTest::newRow("single-colon") << D(":", "", "", ":"); QTest::newRow("single-slash") << D("/", "", "", "/"); QTest::newRow("single-char") << D("a", "", "", "a"); QTest::newRow("relative") << D("./rel", "", "", "rel"); QTest::newRow("qrc") << D(":/test.txt", "", "", ":/test.txt"); QTest::newRow("qrc-no-slash") << D(":test.txt", "", "", ":test.txt"); QTest::newRow("tilde") << D("~/", "", "", QDir::homePath()); QTest::newRow("tilde-with-path") << D("~/foo", "", "", QDir::homePath() + "/foo"); QTest::newRow("tilde-only") << D("~", "", "", QDir::homePath()); QTest::newRow("unc-incomplete") << D("//", "", "", "//"); QTest::newRow("unc-incomplete-only-server") << D("//server", "", "", "//server"); QTest::newRow("unc-incomplete-only-server-2") << D("//server/", "", "", "//server/"); QTest::newRow("unc-server-and-share") << D("//server/share", "", "", "//server/share"); QTest::newRow("unc-server-and-share-2") << D("//server/share/", "", "", "//server/share"); QTest::newRow("unc-full") << D("//server/share/test.txt", "", "", "//server/share/test.txt"); QTest::newRow("unix-root") << D("/", "", "", "/"); QTest::newRow("unix-folder") << D("/tmp", "", "", "/tmp"); QTest::newRow("unix-folder-with-trailing-slash") << D("/tmp/", "", "", "/tmp"); QTest::newRow("windows-root") << D("c:", "", "", "c:"); QTest::newRow("windows-folder") << D("c:/Windows", "", "", "c:/Windows"); QTest::newRow("windows-folder-with-trailing-slash") << D("c:\\Windows\\", "", "", "c:/Windows"); QTest::newRow("windows-folder-slash") << D("C:/Windows", "", "", "C:/Windows"); QTest::newRow("docker-root-url") << D("docker://1234/", "docker", "1234", "/"); QTest::newRow("docker-root-url-special-linux") << D("/__qtc_devices__/docker/1234/", "docker", "1234", "/"); QTest::newRow("docker-root-url-special-win") << D("c:/__qtc_devices__/docker/1234/", "docker", "1234", "/"); QTest::newRow("docker-relative-path") << D("docker://1234/./rel", "docker", "1234", "rel"); QTest::newRow("qtc-dev-linux") << D("/__qtc_devices__", "", "", "/__qtc_devices__"); QTest::newRow("qtc-dev-win") << D("c:/__qtc_devices__", "", "", "c:/__qtc_devices__"); QTest::newRow("qtc-dev-type-linux") << D("/__qtc_devices__/docker", "", "", "/__qtc_devices__/docker"); QTest::newRow("qtc-dev-type-win") << D("c:/__qtc_devices__/docker", "", "", "c:/__qtc_devices__/docker"); QTest::newRow("qtc-dev-type-dev-linux") << D("/__qtc_devices__/docker/1234", "docker", "1234", "/"); QTest::newRow("qtc-dev-type-dev-win") << D("c:/__qtc_devices__/docker/1234", "docker", "1234", "/"); // "Remote Windows" is currently truly not supported. QTest::newRow("cross-os-linux") << D("/__qtc_devices__/docker/1234/c:/test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("cross-os-win") << D("c:/__qtc_devices__/docker/1234/c:/test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("cross-os-unclean-linux") << D("/__qtc_devices__/docker/1234/c:\\test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("cross-os-unclean-win") << D("c:/__qtc_devices__/docker/1234/c:\\test.txt", "docker", "1234", "c:/test.txt", FailEverywhere); QTest::newRow("unc-full-in-docker-linux") << D("/__qtc_devices__/docker/1234//server/share/test.txt", "docker", "1234", "//server/share/test.txt", FailEverywhere); QTest::newRow("unc-full-in-docker-win") << D("c:/__qtc_devices__/docker/1234//server/share/test.txt", "docker", "1234", "//server/share/test.txt", FailEverywhere); QTest::newRow("unc-dos-1") << D("//?/c:", "", "", "c:"); QTest::newRow("unc-dos-com") << D("//./com1", "", "", "//./com1"); } void tst_filepath::fromUserInput() { QFETCH(FromStringData, data); FilePath filePath = FilePath::fromUserInput(data.input); bool expectFail = ((data.expectedPass & FailOnLinux) && !HostOsInfo::isWindowsHost()) || ((data.expectedPass & FailOnWindows) && HostOsInfo::isWindowsHost()); if (expectFail) { QString actual = filePath.scheme() + '|' + filePath.host() + '|' + filePath.path(); QString expected = data.scheme + '|' + data.host + '|' + data.path; QEXPECT_FAIL("", "", Continue); QCOMPARE(actual, expected); return; } QCOMPARE(filePath.scheme(), data.scheme); QCOMPARE(filePath.host(), data.host); QCOMPARE(filePath.path(), data.path); } void tst_filepath::fromToString_data() { QTest::addColumn("scheme"); QTest::addColumn("host"); QTest::addColumn("path"); QTest::addColumn("full"); QTest::newRow("s0") << "" << "" << "" << ""; QTest::newRow("s1") << "" << "" << "/" << "/"; QTest::newRow("s2") << "" << "" << "a/b/c/d" << "a/b/c/d"; QTest::newRow("s3") << "" << "" << "/a/b" << "/a/b"; QTest::newRow("s4") << "docker" << "1234abcdef" << "/bin/ls" << "docker://1234abcdef/bin/ls"; QTest::newRow("s5") << "docker" << "1234" << "/bin/ls" << "docker://1234/bin/ls"; // This is not a proper URL. QTest::newRow("s6") << "docker" << "1234" << "somefile" << "docker://1234/./somefile"; // Local Windows paths: QTest::newRow("w1") << "" << "" << "C:/data" << "C:/data"; QTest::newRow("w2") << "" << "" << "C:/" << "C:/"; QTest::newRow("w3") << "" << "" << "/Global?\?/UNC/host" << "/Global?\?/UNC/host"; QTest::newRow("w4") << "" << "" << "//server/dir/file" << "//server/dir/file"; } void tst_filepath::fromToString() { QFETCH(QString, full); QFETCH(QString, scheme); QFETCH(QString, host); QFETCH(QString, path); FilePath filePath = FilePath::fromString(full); QCOMPARE(filePath.toUrlishString(), full); QCOMPARE(filePath.scheme(), scheme); QCOMPARE(filePath.host(), host); QCOMPARE(filePath.path(), path); FilePath copy = FilePath::fromParts(scheme, host, path); QCOMPARE(copy.toUrlishString(), full); } void tst_filepath::comparison() { QFETCH(QString, left); QFETCH(QString, right); QFETCH(bool, hostSensitive); QFETCH(bool, expected); HostOsInfo::setOverrideFileNameCaseSensitivity(hostSensitive ? Qt::CaseSensitive : Qt::CaseInsensitive); FilePath l = FilePath::fromUserInput(left); FilePath r = FilePath::fromUserInput(right); QCOMPARE(l == r, expected); } void tst_filepath::comparison_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("hostSensitive"); QTest::addColumn("expected"); QTest::newRow("r1") << "Abc" << "abc" << true << false; QTest::newRow("r2") << "Abc" << "abc" << false << true; QTest::newRow("r3") << "x://y/Abc" << "x://y/abc" << true << false; QTest::newRow("r4") << "x://y/Abc" << "x://y/abc" << false << false; QTest::newRow("s1") << "abc" << "abc" << true << true; QTest::newRow("s2") << "abc" << "abc" << false << true; QTest::newRow("s3") << "x://y/abc" << "x://y/abc" << true << true; QTest::newRow("s4") << "x://y/abc" << "x://y/abc" << false << true; } void tst_filepath::linkFromString() { QFETCH(QString, testFile); QFETCH(Utils::FilePath, filePath); QFETCH(int, line); QFETCH(int, column); const Link link = Link::fromString(testFile, true); QCOMPARE(link.targetFilePath, filePath); QCOMPARE(link.target.line, line); QCOMPARE(link.target.column, column); } void tst_filepath::linkFromString_data() { QTest::addColumn("testFile"); QTest::addColumn("filePath"); QTest::addColumn("line"); QTest::addColumn("column"); QTest::newRow("no-line-no-column") << QString("someFile.txt") << FilePath("someFile.txt") << 0 << -1; QTest::newRow(":line-:column") << QString::fromLatin1("/some/path/file.txt:42:3") << FilePath("/some/path/file.txt") << 42 << 2; QTest::newRow("(42) at end") << QString::fromLatin1("/some/path/file.txt(42)") << FilePath("/some/path/file.txt") << 42 << 0; } void tst_filepath::pathAppended() { QFETCH(QString, left); QFETCH(QString, right); QFETCH(QString, expected); const FilePath fleft = FilePath::fromString(left); const FilePath fexpected = FilePath::fromString(expected); const FilePath result = fleft.pathAppended(right); QCOMPARE(result, fexpected); } void tst_filepath::pathAppended_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("expected"); QTest::newRow("p0") << "" << "" << ""; QTest::newRow("p1") << "" << "/" << "/"; QTest::newRow("p2") << "" << "c/" << "c/"; QTest::newRow("p3") << "" << "/d" << "/d"; QTest::newRow("p4") << "" << "c/d" << "c/d"; QTest::newRow("r0") << "/" << "" << "/"; QTest::newRow("r1") << "/" << "/" << "/"; QTest::newRow("r2") << "/" << "c/" << "/c/"; QTest::newRow("r3") << "/" << "/d" << "/d"; QTest::newRow("r4") << "/" << "c/d" << "/c/d"; QTest::newRow("s0") << "/b" << "" << "/b"; QTest::newRow("s1") << "/b" << "/" << "/b/"; QTest::newRow("s2") << "/b" << "c/" << "/b/c/"; QTest::newRow("s3") << "/b" << "/d" << "/b/d"; QTest::newRow("s4") << "/b" << "c/d" << "/b/c/d"; QTest::newRow("t0") << "a/" << "" << "a/"; QTest::newRow("t1") << "a/" << "/" << "a/"; QTest::newRow("t2") << "a/" << "c/" << "a/c/"; QTest::newRow("t3") << "a/" << "/d" << "a/d"; QTest::newRow("t4") << "a/" << "c/d" << "a/c/d"; QTest::newRow("u0") << "a/b" << "" << "a/b"; QTest::newRow("u1") << "a/b" << "/" << "a/b/"; QTest::newRow("u2") << "a/b" << "c/" << "a/b/c/"; QTest::newRow("u3") << "a/b" << "/d" << "a/b/d"; QTest::newRow("u4") << "a/b" << "c/d" << "a/b/c/d"; if (HostOsInfo::isWindowsHost()) { QTest::newRow("win-1") << "c:" << "/a/b" << "c:/a/b"; QTest::newRow("win-2") << "c:/" << "/a/b" << "c:/a/b"; QTest::newRow("win-3") << "c:/" << "a/b" << "c:/a/b"; } } void tst_filepath::resolvePath_string_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << QString() << FilePath(); QTest::newRow("s0") << FilePath("/") << QString("b") << FilePath("/b"); QTest::newRow("s1") << FilePath() << QString("b") << FilePath("b"); QTest::newRow("s2") << FilePath("a") << QString() << FilePath("a"); QTest::newRow("s3") << FilePath("a") << QString("b") << FilePath("a/b"); QTest::newRow("s4") << FilePath("/a") << QString("/b") << FilePath("/b"); QTest::newRow("s5") << FilePath("a") << QString("/b") << FilePath("/b"); QTest::newRow("s6") << FilePath("/a") << QString("b") << FilePath("/a/b"); QTest::newRow("s7") << FilePath("/a") << QString(".") << FilePath("/a"); QTest::newRow("s8") << FilePath("/a") << QString("./b") << FilePath("/a/b"); QTest::newRow("s9") << FilePath("../..") << QString("/b") << FilePath("/b"); QTest::newRow("sa") << FilePath("../..") << QString("b") << FilePath("../../b"); QTest::newRow("sb") << FilePath("a/A") << QString("..") << FilePath("a"); FilePath r = FilePath::fromString("ssh://user@127.0.0.1/tmp/a"); QTest::newRow("r1") << r << QString("bar") << FilePath::fromString("ssh://user@127.0.0.1/tmp/a/bar"); QTest::newRow("r2") << r << QString("..") << FilePath::fromString("ssh://user@127.0.0.1/tmp"); QTest::newRow("r3") << r << QString("../..") << FilePath::fromString("ssh://user@127.0.0.1/"); QTest::newRow("r4") << r << QString("/foo") << FilePath::fromString("ssh://user@127.0.0.1/foo"); } void tst_filepath::resolvePath_string() { QFETCH(FilePath, left); QFETCH(QString, right); QFETCH(FilePath, expected); const FilePath result = left.resolvePath(right); QCOMPARE(result, expected); } void tst_filepath::resolvePath_filepath_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << FilePath() << FilePath(); QTest::newRow("s0") << FilePath("/") << FilePath("b") << FilePath("/b"); QTest::newRow("s1") << FilePath() << FilePath("b") << FilePath("b"); QTest::newRow("s2") << FilePath("a") << FilePath() << FilePath("a"); QTest::newRow("s3") << FilePath("a") << FilePath("b") << FilePath("a/b"); QTest::newRow("s4") << FilePath("/a") << FilePath("/b") << FilePath("/b"); QTest::newRow("s5") << FilePath("a") << FilePath("/b") << FilePath("/b"); QTest::newRow("s6") << FilePath("/a") << FilePath("b") << FilePath("/a/b"); QTest::newRow("s7") << FilePath("/a") << FilePath(".") << FilePath("/a"); QTest::newRow("s8") << FilePath("/a") << FilePath("./b") << FilePath("/a/b"); QTest::newRow("s9") << FilePath("../..") << FilePath("/b") << FilePath("/b"); QTest::newRow("sa") << FilePath("../..") << FilePath("b") << FilePath("../../b"); QTest::newRow("sb") << FilePath("a/A") << FilePath("..") << FilePath("a"); FilePath r = FilePath::fromString("ssh://user@127.0.0.1/tmp/a"); QTest::newRow("r1") << r << FilePath("bar") << FilePath::fromString("ssh://user@127.0.0.1/tmp/a/bar"); QTest::newRow("r2") << r << FilePath("..") << FilePath::fromString("ssh://user@127.0.0.1/tmp"); QTest::newRow("r3") << r << FilePath("../..") << FilePath::fromString("ssh://user@127.0.0.1/"); QTest::newRow("r4") << r << FilePath("/foo") << FilePath::fromString("/foo"); } void tst_filepath::resolvePath_filepath() { QFETCH(FilePath, left); QFETCH(FilePath, right); QFETCH(FilePath, expected); const FilePath result = left.resolvePath(right); QCOMPARE(result, expected); } void tst_filepath::relativeChildPath_data() { QTest::addColumn("parent"); QTest::addColumn("child"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << FilePath() << FilePath(); QTest::newRow("simple-0") << FilePath("/a") << FilePath("/a/b") << FilePath("b"); QTest::newRow("simple-1") << FilePath("/a/") << FilePath("/a/b") << FilePath("b"); QTest::newRow("simple-2") << FilePath("/a") << FilePath("/a/b/c/d/e/f") << FilePath("b/c/d/e/f"); QTest::newRow("not-0") << FilePath("/x") << FilePath("/a/b") << FilePath(); QTest::newRow("not-1") << FilePath("/a/b/c") << FilePath("/a/b") << FilePath(); QTest::newRow("same-0") << FilePath("/a/b") << FilePath("/a/b") << FilePath(); QTest::newRow("same-1") << FilePath("/a/b/") << FilePath("/a/b") << FilePath(); QTest::newRow("same-2") << FilePath("/a/b") << FilePath("/a/b/") << FilePath(); } void tst_filepath::relativeChildPath() { QFETCH(FilePath, parent); QFETCH(FilePath, child); QFETCH(FilePath, expected); const FilePath result = child.relativeChildPath(parent); QCOMPARE(result, expected); } void tst_filepath::asyncLocalCopy() { const FilePath orig = FilePath::fromString(rootPath).pathAppended("x/y/fileToCopy.txt"); QVERIFY(orig.exists()); const FilePath dest = FilePath::fromString(rootPath).pathAppended("x/fileToCopyDest.txt"); bool wasCalled = false; // When QTRY_VERIFY failed, don't call the continuation after we leave this method QObject context; auto afterCopy = [&orig, &dest, &wasCalled](const Result<> &result) { QVERIFY(result); // check existence, size and content QVERIFY(dest.exists()); QCOMPARE(dest.fileSize(), orig.fileSize()); QCOMPARE(dest.fileContents(), orig.fileContents()); wasCalled = true; }; orig.asyncCopy({&context, afterCopy}, dest); QTRY_VERIFY(wasCalled); } void tst_filepath::startsWithDriveLetter_data() { QTest::addColumn("path"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << false; QTest::newRow("simple-win") << FilePath::fromString("c:/a") << true; QTest::newRow("simple-linux") << FilePath::fromString("/c:/a") << false; QTest::newRow("relative") << FilePath("a/b") << false; QTest::newRow("remote-slash") << FilePath::fromString("docker://1234/") << false; QTest::newRow("remote-single-letter") << FilePath::fromString("docker://1234/c") << false; QTest::newRow("remote-drive") << FilePath::fromString("docker://1234/c:") << true; QTest::newRow("remote-invalid-drive") << FilePath::fromString("docker://1234/c:a") << false; QTest::newRow("remote-with-path") << FilePath::fromString("docker://1234/c:/a") << true; QTest::newRow("remote-z") << FilePath::fromString("docker://1234/z:") << true; QTest::newRow("remote-1") << FilePath::fromString("docker://1234/1:") << false; } void tst_filepath::startsWithDriveLetter() { QFETCH(FilePath, path); QFETCH(bool, expected); QCOMPARE(path.startsWithDriveLetter(), expected); } void tst_filepath::withNewMappedPath_data() { QTest::addColumn("path"); QTest::addColumn("templatePath"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << FilePath() << FilePath(); QTest::newRow("same-local") << FilePath("/a/b") << FilePath("/a/b") << FilePath("/a/b"); QTest::newRow("same-docker") << FilePath("docker://1234/a/b") << FilePath("docker://1234/e") << FilePath("docker://1234/a/b"); QTest::newRow("docker-to-local") << FilePath("docker://1234/a/b") << FilePath("/c/d") << FilePath("/a/b"); QTest::newRow("local-to-docker") << FilePath("/a/b") << FilePath("docker://1234/c/d") << FilePath("docker://1234/a/b"); } void tst_filepath::withNewMappedPath() { QFETCH(FilePath, path); QFETCH(FilePath, templatePath); QFETCH(FilePath, expected); QCOMPARE(templatePath.withNewMappedPath(path), expected); } void tst_filepath::stringAppended_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << QString() << FilePath(); QTest::newRow("empty-left") << FilePath() << "a" << FilePath("a"); QTest::newRow("empty-right") << FilePath("a") << QString() << FilePath("a"); QTest::newRow("add-root") << FilePath() << QString("/") << FilePath("/"); QTest::newRow("add-root-and-more") << FilePath() << QString("/test/blah") << FilePath("/test/blah"); QTest::newRow("add-extension") << FilePath::fromString("/a") << QString(".txt") << FilePath("/a.txt"); QTest::newRow("trailing-slash") << FilePath::fromString("/a") << QString("b/") << FilePath("/ab/"); QTest::newRow("slash-trailing-slash") << FilePath::fromString("/a/") << QString("b/") << FilePath("/a/b/"); } void tst_filepath::stringAppended() { QFETCH(FilePath, left); QFETCH(QString, right); QFETCH(FilePath, expected); const FilePath result = left.stringAppended(right); QCOMPARE(expected, result); } void tst_filepath::url() { QFETCH(QString, url); QFETCH(QString, expectedScheme); QFETCH(QString, expectedHost); QFETCH(QString, expectedPath); const FilePath result = FilePath::fromString(url); QCOMPARE(result.scheme(), expectedScheme); QCOMPARE(result.host(), expectedHost); QCOMPARE(result.path(), expectedPath); } void tst_filepath::url_data() { QTest::addColumn("url"); QTest::addColumn("expectedScheme"); QTest::addColumn("expectedHost"); QTest::addColumn("expectedPath"); QTest::newRow("empty") << QString() << QString() << QString() << QString(); QTest::newRow("simple-file") << QString("file:///a/b") << QString("file") << QString() << QString("/a/b"); QTest::newRow("simple-file-root") << QString("file:///") << QString("file") << QString() << QString("/"); QTest::newRow("simple-docker") << QString("docker://1234/a/b") << QString("docker") << QString("1234") << QString("/a/b"); QTest::newRow("simple-ssh") << QString("ssh://user@host/a/b") << QString("ssh") << QString("user@host") << QString("/a/b"); QTest::newRow("simple-ssh-with-port") << QString("ssh://user@host:1234/a/b") << QString("ssh") << QString("user@host:1234") << QString("/a/b"); QTest::newRow("http-qt.io") << QString("http://qt.io") << QString("http") << QString("qt.io") << QString(); QTest::newRow("http-qt.io-index.html") << QString("http://qt.io/index.html") << QString("http") << QString("qt.io") << QString("/index.html"); } void tst_filepath::cleanPath_data() { QTest::addColumn("path"); QTest::addColumn("expected"); QTest::newRow("dot") << "." << "."; QTest::newRow("data0") << "/Users/sam/troll/qt4.0//.." << "/Users/sam/troll"; QTest::newRow("data1") << "/Users/sam////troll/qt4.0//.." << "/Users/sam/troll"; QTest::newRow("data2") << "/" << "/"; QTest::newRow("data2-up") << "/path/.." << "/"; QTest::newRow("data2-above-root") << "/.." << "/.."; QTest::newRow("data3") << QDir::cleanPath("../.") << ".."; QTest::newRow("data4") << QDir::cleanPath("../..") << "../.."; QTest::newRow("data5") << "d:\\a\\bc\\def\\.." << "d:/a/bc"; // QDir/Linux had: "d:\\a\\bc\\def\\.." QTest::newRow("data6") << "d:\\a\\bc\\def\\../../.." << "d:/"; // QDir/Linux had: ".." QTest::newRow("data7") << ".//file1.txt" << "file1.txt"; QTest::newRow("data8") << "/foo/bar/..//file1.txt" << "/foo/file1.txt"; QTest::newRow("data9") << "//" << "//"; // QDir had: "/" QTest::newRow("data10w") << "c:\\" << "c:/"; QTest::newRow("data10l") << "/:/" << "/:"; QTest::newRow("data11") << "//foo//bar" << "//foo/bar"; // QDir/Win had: "//foo/bar" QTest::newRow("data12") << "ab/a/" << "ab/a"; // Path item with length of 2 QTest::newRow("data13w") << "c:/" << "c:/"; QTest::newRow("data13w2") << "c:\\" << "c:/"; //QTest::newRow("data13l") << "c://" << "c:"; // QTest::newRow("data14") << "c://foo" << "c:/foo"; QTest::newRow("data15") << "//c:/foo" << "//c:/foo"; // QDir/Lin had: "/c:/foo"; QTest::newRow("drive-up") << "A:/path/.." << "A:/"; QTest::newRow("drive-above-root") << "A:/.." << "A:/.."; QTest::newRow("unc-server-up") << "//server/path/.." << "//server/"; QTest::newRow("unc-server-above-root") << "//server/.." << "//server/.."; QTest::newRow("longpath") << "\\\\?\\d:\\" << "d:/"; QTest::newRow("longpath-slash") << "//?/d:/" << "d:/"; QTest::newRow("longpath-mixed-slashes") << "//?/d:\\" << "d:/"; QTest::newRow("longpath-mixed-slashes-2") << "\\\\?\\d:/" << "d:/"; QTest::newRow("unc-network-share") << "\\\\?\\UNC\\localhost\\c$\\tmp.txt" << "//localhost/c$/tmp.txt"; QTest::newRow("unc-network-share-slash") << "//?/UNC/localhost/c$/tmp.txt" << "//localhost/c$/tmp.txt"; QTest::newRow("unc-network-share-mixed-slashes") << "//?/UNC/localhost\\c$\\tmp.txt" << "//localhost/c$/tmp.txt"; QTest::newRow("unc-network-share-mixed-slashes-2") << "\\\\?\\UNC\\localhost/c$/tmp.txt" << "//localhost/c$/tmp.txt"; QTest::newRow("QTBUG-23892_0") << "foo/.." << "."; QTest::newRow("QTBUG-23892_1") << "foo/../" << "."; QTest::newRow("QTBUG-3472_0") << "/foo/./bar" << "/foo/bar"; QTest::newRow("QTBUG-3472_1") << "./foo/.." << "."; QTest::newRow("QTBUG-3472_2") << "./foo/../" << "."; QTest::newRow("resource0") << ":/prefix/foo.bar" << ":/prefix/foo.bar"; QTest::newRow("resource1") << ":/prefix/..//prefix/foo.bar" << ":/prefix/foo.bar"; QTest::newRow("ssh") << "ssh://host/prefix/../foo.bar" << "ssh://host/foo.bar"; QTest::newRow("ssh2") << "ssh://host/../foo.bar" << "ssh://host/../foo.bar"; } void tst_filepath::cleanPath() { QFETCH(QString, path); QFETCH(QString, expected); QString cleaned = doCleanPath(path); QCOMPARE(cleaned, expected); } void tst_filepath::isSameFile_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("shouldBeEqual"); QTest::addRow("/==/") << FilePath::fromString("/") << FilePath::fromString("/") << true; QTest::addRow("/!=tmp") << FilePath::fromString("/") << FilePath::fromString(tempDir.path()) << false; QDir dir(tempDir.path()); QVERIFY(touch(dir, "target-file", false)); QFile file(dir.absoluteFilePath("target-file")); if (file.link(dir.absoluteFilePath("source-file"))) { QTest::addRow("real==link") << FilePath::fromString(file.fileName()) << FilePath::fromString(dir.absoluteFilePath("target-file")) << true; } QTest::addRow("/!=non-existing") << FilePath::fromString("/") << FilePath::fromString("/this-path/does-not-exist") << false; QTest::addRow("two-devices") << FilePath::fromString( "docker://boot2qt-raspberrypi4-64:6.5.0/opt/toolchain/sysroots/aarch64-pokysdk-linux/usr/" "bin/aarch64-poky-linux/aarch64-poky-linux-g++") << FilePath::fromString("docker://qt-linux:6/usr/bin/g++") << false; } void tst_filepath::isSameFile() { QFETCH(FilePath, left); QFETCH(FilePath, right); QFETCH(bool, shouldBeEqual); QCOMPARE(left.isSameFile(right), shouldBeEqual); } void tst_filepath::hostSpecialChars_data() { QTest::addColumn("scheme"); QTest::addColumn("host"); QTest::addColumn("path"); QTest::addColumn("expected"); QTest::addRow("slash-in-host") << "device" << "host/name" << "/" << FilePath::fromString("device://host%2fname/"); QTest::addRow("percent-in-host") << "device" << "host%name" << "/" << FilePath::fromString("device://host%25name/"); QTest::addRow("percent-and-slash-in-host") << "device" << "host/%name" << "/" << FilePath::fromString("device://host%2f%25name/"); QTest::addRow("qtc-dev-slash-in-host-linux") << "device" << "host/name" << "/" << FilePath::fromString("/__qtc_devices__/device/host%2fname/"); QTest::addRow("qtc-dev-slash-in-host-windows") << "device" << "host/name" << "/" << FilePath::fromString("c:/__qtc_devices__/device/host%2fname/"); } void tst_filepath::hostSpecialChars() { QFETCH(QString, scheme); QFETCH(QString, host); QFETCH(QString, path); QFETCH(FilePath, expected); FilePath fp; fp.setParts(scheme, host, path); // Check that setParts and fromString give the same result QCOMPARE(fp, expected); QCOMPARE(fp.host(), expected.host()); QCOMPARE(fp.host(), host); QCOMPARE(expected.host(), host); QString toStringExpected = expected.toUrlishString(); QString toStringActual = fp.toUrlishString(); // Check that toString gives the same result QCOMPARE(toStringActual, toStringExpected); // Check that fromString => toString => fromString gives the same result FilePath toFromExpected = FilePath::fromString(expected.toUrlishString()); QCOMPARE(toFromExpected, expected); QCOMPARE(toFromExpected, fp); // Check that setParts => toString => fromString gives the same result FilePath toFromActual = FilePath::fromString(fp.toUrlishString()); QCOMPARE(toFromActual, fp); QCOMPARE(toFromExpected, expected); } void tst_filepath::tmp_data() { QTest::addColumn("templatepath"); QTest::addColumn("expected"); QTest::addRow("empty") << "" << true; QTest::addRow("no-template") << "foo" << true; QTest::addRow("realtive-template") << "my-file-XXXXXXXX" << true; QTest::addRow("absolute-template") << QDir::tempPath() + "/my-file-XXXXXXXX" << true; QTest::addRow("non-existing-dir") << "/this/path/does/not/exist/my-file-XXXXXXXX" << false; QTest::addRow("on-device") << "device://test/./my-file-XXXXXXXX" << true; } void tst_filepath::tmp() { QFETCH(QString, templatepath); QFETCH(bool, expected); FilePath fp = FilePath::fromString(templatepath); const auto result = fp.createTempFile(); QCOMPARE(result.has_value(), expected); if (result.has_value()) { QVERIFY(result->exists()); QVERIFY(result->removeFile()); } } void tst_filepath::sort() { QFETCH(QStringList, input); FilePaths filePaths = Utils::transform(input, &FilePath::fromString); QStringList sorted = input; sorted.sort(); FilePath::sort(filePaths); QStringList sortedPaths = Utils::transform(filePaths, &FilePath::toUrlishString); QCOMPARE(sortedPaths, sorted); } void tst_filepath::lessThan_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("expected"); QTest::newRow("empty") << FilePath() << FilePath() << false; QTest::newRow("simple") << FilePath("/a") << FilePath("/b") << true; QTest::newRow("simple-2") << FilePath("/a") << FilePath("/a") << false; QTest::newRow("simple-3") << FilePath("/b") << FilePath("/a") << false; QTest::newRow("remote-vs-local") << FilePath("docker://1234/a") << FilePath("/a") << false; QTest::newRow("local-vs-remote") << FilePath("/a") << FilePath("docker://1234/a") << true; QTest::newRow("remote-vs-local-2") << FilePath("docker://1234/a") << FilePath("/b") << false; QTest::newRow("local-vs-remote-2") << FilePath("/a") << FilePath("docker://1234/b") << true; } void tst_filepath::lessThan() { QFETCH(FilePath, left); QFETCH(FilePath, right); QFETCH(bool, expected); QCOMPARE(left < right, expected); } void tst_filepath::asQMapKey() { QMap map; map.insert(FilePath::fromString("/Users/mtillmanns/projects/qt/qtc-work/fsengine"), 1); QCOMPARE(map.contains(FilePath::fromString("ssh://marcus@mad-ubuntu-23.local/tmp/untitled")), false); } void tst_filepath::isRootPath_data() { QTest::addColumn("path"); QTest::addColumn("expected"); QTest::newRow("local-root") << FilePath::fromString(QDir::rootPath()) << true; QTest::newRow("local-non-root") << FilePath::fromString(QDir::rootPath() + "x") << false; if (HostOsInfo::isWindowsHost()) { QTest::newRow("remote-windows-root") << FilePath::fromString("device://test/c:/") << true; QTest::newRow("remote-windows-root1") << FilePath::fromString("device://test/c:") << true; QTest::newRow("remote-windows-not-root") << FilePath::fromString("device://test/c:/x") << false; QTest::newRow("local-c-drive") << FilePath::fromString("c:/") << true; QTest::newRow("local-d-drive") << FilePath::fromString("d:/") << true; QTest::newRow("local-c-drive-noslash") << FilePath::fromString("c:") << false; QTest::newRow("local-d-drive-noslash") << FilePath::fromString("d:") << false; QTest::newRow("unc-root-and-share") << FilePath::fromString("//server/share") << false; QTest::newRow("unc-root") << FilePath::fromString("//server") << true; } else { QTest::newRow("remote-root") << FilePath::fromString("device://test/") << true; QTest::newRow("remote-path") << FilePath::fromString("device://test/x") << false; } } void tst_filepath::isRootPath() { QFETCH(FilePath, path); QFETCH(bool, expected); QCOMPARE(path.isRootPath(), expected); } void tst_filepath::sort_data() { QTest::addColumn("input"); QTest::addRow("empty") << QStringList{}; QTest::addRow("one") << QStringList{"foo"}; QTest::addRow("two") << QStringList{"foo", "bar"}; QTest::addRow("three") << QStringList{"foo", "bar", "baz"}; QTest::addRow("one-absolute") << QStringList{"/foo"}; QTest::addRow("two-absolute") << QStringList{"/foo", "/bar"}; QTest::addRow("one-relative") << QStringList{"foo"}; QTest::addRow("one-absolute-one-relative") << QStringList{"/foo", "bar"}; QTest::addRow("host") << QStringList{"ssh://test/blah", "ssh://gulp/blah", "ssh://zzz/blah"}; QTest::addRow("scheme") << QStringList{"ssh://test/blah", "ssh://gulp/blah", "ssh://zzz/blah", "aaa://gulp/blah", "xyz://test/blah"}; QTest::addRow("others") << QStringList{"a://a//a", "a://b//a", "a://a//b", "a://b//b", "b://b//b"}; QTest::addRow("others-reversed") << QStringList{"b://b//b", "a://b//b", "a://a//b", "a://b//a", "a://a//a"}; } void tst_filepath::makeTemporaryFile() { FilePath tmpFilePath; // Test auto remove { const FilePath tmplate = FilePath::fromUserInput(QDir::tempPath()) / "test-auto-remove-XXXXXX.txt"; auto tmpFile = TemporaryFilePath::create(tmplate); QVERIFY(tmpFile); QVERIFY(!(*tmpFile)->templatePath().exists()); QVERIFY((*tmpFile)->filePath().exists()); tmpFilePath = (*tmpFile)->filePath(); } QVERIFY(!tmpFilePath.exists()); // Check !autoRemove { const FilePath tmplate = FilePath::fromUserInput(QDir::tempPath()) / "test-no-auto-remove-XXXXXX.txt"; auto tmpFile = TemporaryFilePath::create(tmplate); QVERIFY(tmpFile); (*tmpFile)->setAutoRemove(false); QVERIFY(!(*tmpFile)->templatePath().exists()); QVERIFY((*tmpFile)->filePath().exists()); tmpFilePath = (*tmpFile)->filePath(); } QVERIFY(tmpFilePath.exists()); QVERIFY(tmpFilePath.removeFile()); // Check invalid filename { const FilePath tmplate = FilePath::fromUserInput("/Some/non/existing/path") / "test-invalid-filename-XXXXXX"; auto tmpFile = TemporaryFilePath::create(tmplate); QVERIFY(!tmpFile); } } void tst_filepath::isRelativePath_data() { QTest::addColumn("path"); QTest::addColumn("expected"); QTest::newRow("empty") << "" << true; QTest::newRow("root") << "/" << false; QTest::newRow("relative") << "foo" << true; QTest::newRow("relative-path") << "foo/bar" << true; QTest::newRow("absolute") << "/foo" << false; QTest::newRow("absolute-path") << "/foo/bar" << false; QTest::newRow("remote") << "device://host/foo" << false; QTest::newRow("remote-path") << "device://host/foo/bar" << false; QTest::newRow("windows-current-dir") << "c:" << true; QTest::newRow("windows-path") << "c:/" << false; QTest::newRow("windows-path-with-dir") << "c:/foo" << false; QTest::newRow("windows-remote-current-dir") << "device://host/C:" << true; QTest::newRow("windows-remote-path") << "device://host/C:/" << false; QTest::newRow("windows-remote-path-with-dir") << "device://host/C:/foo" << false; QTest::newRow("qrc-absolute-path-variant1") << ":/foo/bar" << false; QTest::newRow("qrc-absolute-path-variant2") << ":foo/bar" << false; } void tst_filepath::isRelativePath() { QFETCH(QString, path); QFETCH(bool, expected); QCOMPARE(FilePath::fromUserInput(path).isRelativePath(), expected); } void tst_filepath::dontBreakPathOnWierdWindowsPaths() { FilePath path = FilePath::fromString( "device://host/./C:/Users/johndoe/Documents/" "build-iartest-IAR-Debugx/Debug_IAR_55df6f02d5b3d06d/iartest.a152245e/iartest.out"); QCOMPARE( path.toUrlishString(), "device://host/C:/Users/johndoe/Documents/" "build-iartest-IAR-Debugx/Debug_IAR_55df6f02d5b3d06d/iartest.a152245e/iartest.out"); FilePath pathWithBackslash = FilePath::fromUserInput( "device://host/C:\\Users\\johndoe\\Documents\\test.elf"); QCOMPARE(pathWithBackslash.toUrlishString(), "device://host/C:/Users/johndoe/Documents/test.elf"); QCOMPARE(pathWithBackslash.path(), "C:/Users/johndoe/Documents/test.elf"); const FilePath bin = FilePath::fromString(pathWithBackslash.path()); QCOMPARE(bin.toUrlishString(), "C:/Users/johndoe/Documents/test.elf"); // Make sure the optimization still works FilePath path2 = FilePath::fromString("/./"); QCOMPARE(path2.toUrlishString(), ""); // Make sure unix paths are not affected FilePath path3 = FilePath::fromString("/./foo/bar"); QCOMPARE(path3.toUrlishString(), "foo/bar"); // Make sure unix paths with device also work FilePath path4 = FilePath::fromString("device://host/./foo/bar"); QCOMPARE(path4.toUrlishString(), "device://host/./foo/bar"); } void tst_filepath::pathComponents_data() { QTest::addColumn("path"); QTest::addColumn("expected"); QTest::newRow("empty") << "" << QStringList{}; QTest::newRow("root") << "/" << QStringList{"/"}; QTest::newRow("relative") << "foo" << QStringList{"foo"}; QTest::newRow("relative-path") << "foo/bar" << QStringList{"foo", "bar"}; QTest::newRow("absolute") << "/foo" << QStringList{"/", "foo"}; QTest::newRow("absolute-path") << "/foo/bar" << QStringList{"/", "foo", "bar"}; QTest::newRow("remote") << "device://host/foo" << QStringList{"/", "foo"}; QTest::newRow("remote-path") << "device://host/foo/bar" << QStringList{"/", "foo", "bar"}; QTest::newRow("remote-relative") << "device://host/./foo" << QStringList{"foo"}; QTest::newRow("windows-current-dir") << "c:" << QStringList{"c:"}; QTest::newRow("single-letter") << "c" << QStringList{"c"}; QTest::newRow("single-letter-path") << "c/b" << QStringList{"c", "b"}; QTest::newRow("single-letter-path-with-root") << "/c/b" << QStringList{"/", "c", "b"}; if (HostOsInfo::isWindowsHost()) { QTest::newRow("cwd-windows-path") << "c:foo" << QStringList{"c:", "foo"}; QTest::newRow("windows-path") << "c:/" << QStringList{"c:", "/"}; QTest::newRow("windows-path-2") << "c:/test" << QStringList{"c:", "/", "test"}; QTest::newRow("single-letter-path-with-windows-root") << "c:/a/b/c" << QStringList{"c:", "/", "a", "b", "c"}; QTest::newRow("windows-path-with-dir") << "c:/foo" << QStringList{"c:", "/", "foo"}; } else { QTest::newRow("cwd-windows-path") << "c:foo" << QStringList{"c:foo"}; QTest::newRow("windows-path") << "c:/" << QStringList{"c:"}; QTest::newRow("windows-path-2") << "c:/test" << QStringList{"c:", "test"}; QTest::newRow("single-letter-path-with-windows-root") << "c:/a/b/c" << QStringList{"c:", "a", "b", "c"}; QTest::newRow("windows-path-with-dir") << "c:/foo" << QStringList{"c:", "foo"}; } } void tst_filepath::symLinks() { const FilePath orig = FilePath::fromString(rootPath).pathAppended("x/y/fileToCopy.txt"); QVERIFY(orig.exists()); const FilePath link = FilePath::fromString(rootPath).pathAppended("x/fileToCopySymLink.txt"); const Result<> res = orig.createSymLink(link); if (HostOsInfo::isWindowsHost() && !res) { QSKIP("Creating symbolic links requires special privileges on Windows: " "Local group policy editor (gpedit.msc) " "computer configuration > Windows settings > security settings " "> local policy > user rights > creating symbolic links"); } QVERIFY(link.isSymLink()); QCOMPARE(link.symLinkTarget(), orig); } void tst_filepath::resolveSymLinks() { // relative link chain to existing file const FilePath orig1 = FilePath::fromString(rootPath).pathAppended("a/file3.txt"); QVERIFY(orig1.exists()); const FilePath link1 = FilePath::fromString(rootPath).pathAppended("x/linktest1.txt"); const Result<> res1 = FilePath::fromString("../a/file3.txt").createSymLink(link1); if (HostOsInfo::isWindowsHost() && !res1) { QSKIP("Creating symbolic links requires special privileges on Windows: " "Local group policy editor (gpedit.msc) " "computer configuration > Windows settings > security settings " "> local policy > user rights > creating symbolic links"); } QVERIFY_RESULT(res1); QVERIFY(link1.isSymLink()); QCOMPARE(link1.symLinkTarget(), orig1); const FilePath link2 = FilePath::fromString(rootPath).pathAppended("x/linktest2.txt"); const Result<> res2 = FilePath::fromString("linktest1.txt").createSymLink(link2); QVERIFY_RESULT(res2); QVERIFY(link2.isSymLink()); QCOMPARE(link2.symLinkTarget(), link1); QCOMPARE(link2.resolveSymlinks(), orig1); // relative link chain to non-existing file const FilePath orig2 = FilePath::fromString(rootPath).pathAppended("a/broken.txt"); QVERIFY(!orig2.exists()); const FilePath link3 = FilePath::fromString(rootPath).pathAppended("x/linktest3.txt"); const Result<> res3 = FilePath::fromString("../a/broken.txt").createSymLink(link3); QVERIFY_RESULT(res3); QVERIFY(link3.isSymLink()); QCOMPARE(link3.symLinkTarget(), orig2); const FilePath link4 = FilePath::fromString(rootPath).pathAppended("x/linktest4.txt"); const Result<> res4 = FilePath::fromString("linktest3.txt").createSymLink(link4); QVERIFY_RESULT(res4); QVERIFY(link4.isSymLink()); QCOMPARE(link4.symLinkTarget(), link3); QCOMPARE(link4.resolveSymlinks(), orig2); // relative links in directory links // currently: // /a/file3.txt // /x/linktest1.txt -> ../a/file3.txt // if now: // /a/dirlink -> ../x // then resolution should happen like this: // /a/dirlink/linktest1.txt -> /a/file3.txt const FilePath dirlink = FilePath::fromString(rootPath).pathAppended("a/dirlink"); const Result<> res5 = FilePath::fromString("../x").createSymLink(dirlink); QVERIFY_RESULT(res5); QVERIFY(dirlink.isSymLink()); QCOMPARE(dirlink.symLinkTarget(), FilePath::fromString(rootPath).pathAppended("x")); QCOMPARE(dirlink.pathAppended("linktest1.txt").resolveSymlinks(), orig1); } void tst_filepath::ensureWritableDirectory() { const FilePath dir = FilePath::fromString(rootPath).pathAppended("x/y/writableDir"); QVERIFY(!dir.exists()); Result<> res = dir.ensureWritableDir(); QVERIFY_RESULT(res); QVERIFY(dir.exists()); QVERIFY(dir.isWritableDir()); const FilePath notdir = FilePath::fromString(rootPath).pathAppended("x/y/notADirectory.txt"); QVERIFY(!notdir.exists()); QVERIFY_RESULT(notdir.writeFileContents({})); QVERIFY(notdir.exists()); res = notdir.ensureWritableDir(); QVERIFY(!res); QVERIFY(notdir.isFile()); } void tst_filepath::ensureWritableDirectoryPermissions() { if (HostOsInfo::isWindowsHost()) QSKIP("Permissions are not supported on Windows"); const FilePath dir2 = FilePath::fromString(rootPath).pathAppended("x/y/initiallyNotWritableDir"); QVERIFY(!dir2.exists()); const Result<> res = dir2.ensureWritableDir(); QVERIFY_RESULT(res); QVERIFY(dir2.exists()); QVERIFY(dir2.isWritableDir()); dir2.setPermissions(QFile::Permissions(QFile::ReadOwner | QFile::ReadGroup | QFile::ReadOther)); QVERIFY(dir2.exists()); QVERIFY(!dir2.isWritableDir()); QVERIFY(!dir2.ensureWritableDir()); QVERIFY(dir2.exists()); QVERIFY(!dir2.isWritableDir()); } void tst_filepath::searchHereAndInParents() { const FilePath dir = FilePath::fromString(rootPath).pathAppended("a/b/c/d"); QVERIFY(dir.isDir()); // Do not find (not reachable by going up from here). const FilePath file2 = dir.searchHereAndInParents("file2.txt", QDir::Files); QVERIFY2(file2.isEmpty(), qPrintable(file2.toUserOutput())); // Do not find (wrong type). const FilePath file1Dir = dir.searchHereAndInParents("file1.txt", QDir::Dirs); QVERIFY2(file1Dir.isEmpty(), qPrintable(file1Dir.toUserOutput())); // Find in same dir. const FilePath file1 = dir.searchHereAndInParents("file1.txt", QDir::Files); QCOMPARE(file1, dir.pathAppended("file1.txt")); // Find in some parent dir. const FilePath file3 = file1.searchHereAndInParents("file3.txt", QDir::Files); QCOMPARE(file3, FilePath::fromString(rootPath).pathAppended("a/file3.txt")); } void tst_filepath::pathComponents() { QFETCH(QString, path); QFETCH(QStringList, expected); const auto components = Utils::transform(FilePath::fromString(path).pathComponents(), &QStringView::toString); QCOMPARE(components, expected); } void tst_filepath::parentsWithDevice() { const FilePath path = FilePath::fromUserInput("test://test/a/b/c"); const PathAndParents parentPaths(path); auto it = std::begin(parentPaths); QCOMPARE(*it, FilePath::fromUserInput("test://test/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("test://test/a/b")); ++it; QCOMPARE(*it, FilePath::fromUserInput("test://test/a")); ++it; QCOMPARE(*it, FilePath::fromUserInput("test://test/")); ++it; QCOMPARE(it, std::end(parentPaths)); } void tst_filepath::parentsWithDrive() { const FilePath path = FilePath::fromUserInput("C:/a/b/c"); const PathAndParents parentPaths(path); auto it = std::begin(parentPaths); QCOMPARE(*it, FilePath::fromUserInput("C:/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("C:/a/b")); ++it; QCOMPARE(*it, FilePath::fromUserInput("C:/a")); ++it; QCOMPARE(*it, FilePath::fromUserInput("C:/")); ++it; QCOMPARE(it, std::end(parentPaths)); } void tst_filepath::parentsWithUncPath() { const FilePath path = FilePath::fromUserInput("//server/share/a/b/c"); const PathAndParents parentPaths(path); auto it = std::begin(parentPaths); QCOMPARE(*it, FilePath::fromUserInput("//server/share/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("//server/share/a/b")); ++it; QCOMPARE(*it, FilePath::fromUserInput("//server/share/a")); ++it; QCOMPARE(*it, FilePath::fromUserInput("//server/share")); ++it; QCOMPARE(*it, FilePath::fromUserInput("//server/")); ++it; QCOMPARE(it, std::end(parentPaths)); } void tst_filepath::emptyParents() { const FilePath path = FilePath::fromUserInput(""); const PathAndParents parentPaths(path); auto it = std::begin(parentPaths); QCOMPARE(it, std::end(parentPaths)); } void tst_filepath::parents() { const FilePath path = FilePath::fromUserInput("/a/b/c"); const PathAndParents parentPaths(path); auto it = std::begin(parentPaths); QCOMPARE(*it, FilePath::fromUserInput("/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/")); ++it; QCOMPARE(it, std::end(parentPaths)); } void tst_filepath::parentsWithLastPath() { const FilePath path = FilePath::fromUserInput("/a/b/c/d"); const PathAndParents parentPaths(path, FilePath::fromUserInput("/a/b")); auto it = std::begin(parentPaths); QCOMPARE(*it, FilePath::fromUserInput("/a/b/c/d")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b")); ++it; QCOMPARE(it, std::end(parentPaths)); const PathAndParents parentPaths2(path, path); it = std::begin(parentPaths2); QCOMPARE(*it, FilePath::fromUserInput("/a/b/c/d")); ++it; QCOMPARE(it, std::end(parentPaths2)); // Specifying a path that is not a parent of the given path // should fall back to iterating until the root. ignoreSoftAssert(); const PathAndParents parentPaths3(path, FilePath::fromUserInput("/x")); it = std::begin(parentPaths3); QCOMPARE(*it, FilePath::fromUserInput("/a/b/c/d")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/")); ++it; ignoreSoftAssert(); QCOMPARE(it, std::end(parentPaths3)); const PathAndParents emptyLast(path, FilePath()); it = std::begin(emptyLast); QCOMPARE(*it, FilePath::fromUserInput("/a/b/c/d")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b/c")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a/b")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/a")); ++it; QCOMPARE(*it, FilePath::fromUserInput("/")); ++it; QCOMPARE(it, std::end(emptyLast)); } void tst_filepath::exists() { const Result tmpPath = FilePath().tmpDir(); QVERIFY_RESULT(tmpPath); const FilePath pattern = tmpPath.value() / "test.XXXXXXXXXXX"; const Result resultPath = pattern.createTempFile(); QVERIFY_RESULT(resultPath); QVERIFY(resultPath->exists()); QVERIFY_RESULT(resultPath->removeFile()); QVERIFY(!resultPath->exists()); } void tst_filepath::isNewerThan() { const QDateTime time = QDateTime::currentDateTime().addSecs(-1); const Result tmpPath = FilePath().tmpDir(); QVERIFY_RESULT(tmpPath); const FilePath pattern = (tmpPath.value() / "test.XXXXXXXXXXX"); const Result resultPath = pattern.createTempFile(); QVERIFY_RESULT(resultPath); QVERIFY(resultPath->isNewerThan(time)); QVERIFY(!resultPath->isNewerThan(QDateTime::currentDateTime())); QVERIFY(!resultPath->isNewerThan(resultPath->lastModified())); } // QSignalSpy does not work with signals from threads, because it uses a direct connection // so add a QObject in between class Spy : public QObject { Q_OBJECT public: Spy(FilePathWatcher *watcher) : signalSpy(this, &Spy::trigger) { QObject::connect(watcher, &FilePathWatcher::pathChanged, this, &Spy::trigger); } bool wait(int millis) { return signalSpy.wait(millis); } void clear() { signalSpy.clear(); } qsizetype count() { return signalSpy.count(); } signals: void trigger(); private: QSignalSpy signalSpy; }; void tst_filepath::watch() { const FilePath rootDir = FilePath::fromString(rootPath); const auto fileName = [&rootDir](int i) { return rootDir.pathAppended("watchfile" + QString::number(i) + ".txt"); }; const auto createFile = [fileName](int i) { const auto filePath = fileName(i); filePath.writeFileContents("test"); return filePath; }; FilePaths watchFiles = Utils::transform>({0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10}, createFile); // Add one that does not exist to make sure the rest still works. watchFiles.append(rootDir.pathAppended("nonexistingfile.txt")); std::vector>> firstWatches = watchFiles.watch(); QVERIFY(!firstWatches.back().has_value()); firstWatches.pop_back(); for (const auto &watchResult : firstWatches) QVERIFY_RESULT(watchResult); // Verify that modifying one of the files triggers its watcher Spy spy1(firstWatches[0].value().get()); QVERIFY_RESULT(fileName(0).writeFileContents("test 1")); QVERIFY(spy1.wait(3000)); QCOMPARE(spy1.count(), 1); spy1.clear(); // Make sure triggering a different file does not trigger the watcher of the first file QVERIFY_RESULT(fileName(1).writeFileContents("test 2")); QVERIFY_RESULT(fileName(0).writeFileContents("test 1")); QVERIFY(spy1.wait(3000)); QCOMPARE(spy1.count(), 1); spy1.clear(); // Add another watcher on the first file const std::vector>> secondWatches = FilePaths{fileName(0)}.watch(); for (const auto &watchResult : secondWatches) QVERIFY_RESULT(watchResult); // Make sure that both watchers are triggered when the file changes Spy spy2(secondWatches[0].value().get()); QVERIFY_RESULT(fileName(0).writeFileContents("test 2")); QVERIFY(spy2.wait(3000)); QCOMPARE(spy2.count(), 1); QCOMPARE(spy1.count(), 1); spy1.clear(); spy2.clear(); // Make sure that removing the first watcher(s) leaves the second working firstWatches.clear(); QVERIFY_RESULT(fileName(0).writeFileContents("test 3")); QVERIFY(spy2.wait(3000)); QCOMPARE(spy2.count(), 1); QCOMPARE(spy1.count(), 0); spy1.clear(); spy2.clear(); // Test replacement like it is done for SaveFile const FilePath saveFile = rootDir / "watchfile.savefile"; saveFile.writeFileContents("test for save"); fileName(0).removeFile(); QVERIFY_RESULT(saveFile.renameFile(fileName(0))); QVERIFY(spy2.wait(3000)); QCOMPARE(spy2.count(), 1); spy2.clear(); // Check that we still are notified for further changes QVERIFY_RESULT(fileName(0).writeFileContents("change after replace")); QVERIFY(spy2.wait(3000)); QCOMPARE(spy2.count(), 1); } } // Utils QTEST_GUILESS_MAIN(Utils::tst_filepath) #include "tst_filepath.moc"