X Tutup
/* * Cppcheck - A tool for static C/C++ code analysis * Copyright (C) 2007-2026 Cppcheck team. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "path.h" #include "fixture.h" #include "helpers.h" #include "standards.h" #include #include #include #include #include #ifndef _WIN32 #include #endif class TestPath : public TestFixture { public: TestPath() : TestFixture("TestPath") {} private: void run() override { TEST_CASE(removeQuotationMarks); TEST_CASE(acceptFile); TEST_CASE(getCurrentPath); TEST_CASE(getCurrentExecutablePath); TEST_CASE(isAbsolute); TEST_CASE(getRelative); TEST_CASE(get_path_from_filename); TEST_CASE(join); TEST_CASE(isDirectory); TEST_CASE(isFile); TEST_CASE(sameFileName); TEST_CASE(getFilenameExtension); TEST_CASE(identify); TEST_CASE(identifyWithCppProbe); TEST_CASE(is_header); TEST_CASE(simplifyPath); TEST_CASE(getAbsolutePath); TEST_CASE(exists); TEST_CASE(fromNativeSeparators); } void removeQuotationMarks() const { // Path::removeQuotationMarks() ASSERT_EQUALS("index.cpp", Path::removeQuotationMarks("index.cpp")); ASSERT_EQUALS("index.cpp", Path::removeQuotationMarks("\"index.cpp")); ASSERT_EQUALS("index.cpp", Path::removeQuotationMarks("index.cpp\"")); ASSERT_EQUALS("index.cpp", Path::removeQuotationMarks("\"index.cpp\"")); ASSERT_EQUALS("path to/index.cpp", Path::removeQuotationMarks("\"path to\"/index.cpp")); ASSERT_EQUALS("path to/index.cpp", Path::removeQuotationMarks("\"path to/index.cpp\"")); ASSERT_EQUALS("the/path to/index.cpp", Path::removeQuotationMarks("the/\"path to\"/index.cpp")); ASSERT_EQUALS("the/path to/index.cpp", Path::removeQuotationMarks("\"the/path to/index.cpp\"")); } void acceptFile() const { ASSERT(Path::acceptFile("index.c")); ASSERT(Path::acceptFile("index.cpp")); ASSERT(Path::acceptFile("index.invalid.cpp")); ASSERT(Path::acceptFile("index.invalid.Cpp")); ASSERT(Path::acceptFile("index.invalid.C")); ASSERT(Path::acceptFile("index.invalid.C++")); ASSERT(Path::acceptFile("index.")==false); ASSERT(Path::acceptFile("index")==false); ASSERT(Path::acceptFile("")==false); ASSERT(Path::acceptFile("C")==false); { Standards::Language lang; ASSERT(Path::acceptFile("index.c", &lang)); ASSERT_EQUALS_ENUM(Standards::Language::C, lang); } { Standards::Language lang; ASSERT(Path::acceptFile("index.cpp", &lang)); ASSERT_EQUALS_ENUM(Standards::Language::CPP, lang); } // don't accept any headers ASSERT_EQUALS(false, Path::acceptFile("index.h")); ASSERT_EQUALS(false, Path::acceptFile("index.hpp")); const std::set extra = { ".extra", ".header" }; ASSERT(Path::acceptFile("index.c", extra)); ASSERT(Path::acceptFile("index.cpp", extra)); ASSERT(Path::acceptFile("index.extra", extra)); ASSERT(Path::acceptFile("index.header", extra)); ASSERT(Path::acceptFile("index.h", extra)==false); ASSERT(Path::acceptFile("index.hpp", extra)==false); { Standards::Language lang; ASSERT(Path::acceptFile("index.c", extra, &lang)); ASSERT_EQUALS_ENUM(Standards::Language::C, lang); } { Standards::Language lang; ASSERT(Path::acceptFile("index.cpp", extra, &lang)); ASSERT_EQUALS_ENUM(Standards::Language::CPP, lang); } { Standards::Language lang; ASSERT(Path::acceptFile("index.extra", extra, &lang)); ASSERT_EQUALS_ENUM(Standards::Language::None, lang); } } void getCurrentPath() const { ASSERT_EQUALS(true, Path::isAbsolute(Path::getCurrentPath())); } void getCurrentExecutablePath() const { ASSERT_EQUALS(false, Path::getCurrentExecutablePath("").empty()); } void isAbsolute() const { #ifdef _WIN32 ASSERT_EQUALS(true, Path::isAbsolute("C:\\foo\\bar")); ASSERT_EQUALS(true, Path::isAbsolute("C:/foo/bar")); ASSERT_EQUALS(true, Path::isAbsolute("\\\\foo\\bar")); ASSERT_EQUALS(false, Path::isAbsolute("foo\\bar")); ASSERT_EQUALS(false, Path::isAbsolute("foo/bar")); ASSERT_EQUALS(false, Path::isAbsolute("foo.cpp")); ASSERT_EQUALS(false, Path::isAbsolute("C:foo.cpp")); ASSERT_EQUALS(false, Path::isAbsolute("C:foo\\bar.cpp")); ASSERT_EQUALS(false, Path::isAbsolute("bar.cpp")); TODO_ASSERT_EQUALS(true, false, Path::isAbsolute("\\")); #else ASSERT_EQUALS(true, Path::isAbsolute("/foo/bar")); ASSERT_EQUALS(true, Path::isAbsolute("/")); ASSERT_EQUALS(false, Path::isAbsolute("foo/bar")); ASSERT_EQUALS(false, Path::isAbsolute("foo.cpp")); #endif } void getRelative() const { const std::vector basePaths = { "", // Don't crash with empty paths "C:/foo", "C:/bar/", "C:/test.cpp" }; ASSERT_EQUALS("x.c", Path::getRelativePath("C:/foo/x.c", basePaths)); ASSERT_EQUALS("y.c", Path::getRelativePath("C:/bar/y.c", basePaths)); ASSERT_EQUALS("foo/y.c", Path::getRelativePath("C:/bar/foo/y.c", basePaths)); ASSERT_EQUALS("C:/test.cpp", Path::getRelativePath("C:/test.cpp", basePaths)); ASSERT_EQUALS("C:/foobar/test.cpp", Path::getRelativePath("C:/foobar/test.cpp", basePaths)); } void get_path_from_filename() const { ASSERT_EQUALS("", Path::getPathFromFilename("index.h")); ASSERT_EQUALS("/tmp/", Path::getPathFromFilename("/tmp/index.h")); ASSERT_EQUALS("a/b/c/", Path::getPathFromFilename("a/b/c/index.h")); ASSERT_EQUALS("a/b/c/", Path::getPathFromFilename("a/b/c/")); } void join() const { ASSERT_EQUALS("", Path::join("", "")); ASSERT_EQUALS("b", Path::join("", "b")); ASSERT_EQUALS("/b", Path::join("", "/b")); ASSERT_EQUALS("/b", Path::join("", "\\b")); ASSERT_EQUALS("a", Path::join("a", "")); ASSERT_EQUALS("a/b", Path::join("a", "b")); ASSERT_EQUALS("/b", Path::join("a", "/b")); ASSERT_EQUALS("/b", Path::join("a", "\\b")); ASSERT_EQUALS("a/", Path::join("a/", "")); ASSERT_EQUALS("a/b", Path::join("a/", "b")); ASSERT_EQUALS("/b", Path::join("a/", "/b")); ASSERT_EQUALS("/b", Path::join("a/", "\\b")); ASSERT_EQUALS("a/", Path::join("a\\", "")); ASSERT_EQUALS("a/b", Path::join("a\\", "b")); ASSERT_EQUALS("/b", Path::join("a\\", "/b")); ASSERT_EQUALS("/b", Path::join("a\\", "\\b")); // TODO: how to absolute Windows path in path2? //ASSERT_EQUALS("", Path::join("a", "s:/b")); //ASSERT_EQUALS("", Path::join("S:\\a", "S:/b")); //ASSERT_EQUALS("", Path::join("S:\\a", "S:\\b")); //ASSERT_EQUALS("", Path::join("S:\\a", "/b")); //ASSERT_EQUALS("", Path::join("S:/a", "S:/b")); //ASSERT_EQUALS("", Path::join("S:/a", "S:\\b")); //ASSERT_EQUALS("", Path::join("S:/a", "/b")); } void isDirectory() const { ScopedFile file("testpath.txt", "", "testpath"); ScopedFile file2("testpath2.txt", ""); ASSERT_EQUALS(false, Path::isDirectory("testpath.txt")); ASSERT_EQUALS(true, Path::isDirectory("testpath")); ASSERT_EQUALS(false, Path::isDirectory("testpath/testpath.txt")); ASSERT_EQUALS(false, Path::isDirectory("testpath2.txt")); } void isFile() const { ScopedFile file("testpath.txt", "", "testpath"); ScopedFile file2("testpath2.txt", ""); ASSERT_EQUALS(false, Path::isFile("testpath")); ASSERT_EQUALS(false, Path::isFile("testpath.txt")); ASSERT_EQUALS(true, Path::isFile("testpath/testpath.txt")); ASSERT_EQUALS(true, Path::isFile("testpath2.txt")); } void sameFileName() const { ASSERT(Path::sameFileName("test", "test")); // case sensitivity cases #if defined(_WIN32) || (defined(__APPLE__) && defined(__MACH__)) ASSERT(Path::sameFileName("test", "Test")); ASSERT(Path::sameFileName("test", "TesT")); ASSERT(Path::sameFileName("test.h", "test.H")); ASSERT(Path::sameFileName("test.hh", "test.Hh")); ASSERT(Path::sameFileName("test.hh", "test.hH")); #else ASSERT(!Path::sameFileName("test", "Test")); ASSERT(!Path::sameFileName("test", "TesT")); ASSERT(!Path::sameFileName("test.h", "test.H")); ASSERT(!Path::sameFileName("test.hh", "test.Hh")); ASSERT(!Path::sameFileName("test.hh", "test.hH")); #endif } void getFilenameExtension() const { ASSERT_EQUALS("", Path::getFilenameExtension("test")); ASSERT_EQUALS("", Path::getFilenameExtension("Test")); ASSERT_EQUALS(".h", Path::getFilenameExtension("test.h")); ASSERT_EQUALS(".h", Path::getFilenameExtension("Test.h")); ASSERT_EQUALS("", Path::getFilenameExtension("test", true)); ASSERT_EQUALS("", Path::getFilenameExtension("Test", true)); ASSERT_EQUALS(".h", Path::getFilenameExtension("test.h", true)); ASSERT_EQUALS(".h", Path::getFilenameExtension("Test.h", true)); // case sensitivity cases #if defined(_WIN32) || (defined(__APPLE__) && defined(__MACH__)) ASSERT_EQUALS(".h", Path::getFilenameExtension("test.H")); ASSERT_EQUALS(".hh", Path::getFilenameExtension("test.Hh")); ASSERT_EQUALS(".hh", Path::getFilenameExtension("test.hH")); ASSERT_EQUALS(".h", Path::getFilenameExtension("test.H", true)); ASSERT_EQUALS(".hh", Path::getFilenameExtension("test.Hh", true)); ASSERT_EQUALS(".hh", Path::getFilenameExtension("test.hH", true)); #else ASSERT_EQUALS(".H", Path::getFilenameExtension("test.H")); ASSERT_EQUALS(".Hh", Path::getFilenameExtension("test.Hh")); ASSERT_EQUALS(".hH", Path::getFilenameExtension("test.hH")); ASSERT_EQUALS(".h", Path::getFilenameExtension("test.H", true)); ASSERT_EQUALS(".hh", Path::getFilenameExtension("test.Hh", true)); ASSERT_EQUALS(".hh", Path::getFilenameExtension("test.hH", true)); #endif } void identify() const { Standards::Language lang; bool header; ASSERT_EQUALS(Standards::Language::None, Path::identify("", false)); ASSERT_EQUALS(Standards::Language::None, Path::identify("c", false)); ASSERT_EQUALS(Standards::Language::None, Path::identify("cpp", false)); ASSERT_EQUALS(Standards::Language::None, Path::identify("h", false)); ASSERT_EQUALS(Standards::Language::None, Path::identify("hpp", false)); // TODO: what about files starting with a "."? //ASSERT_EQUALS(Standards::Language::None, Path::identify(".c", false)); //ASSERT_EQUALS(Standards::Language::None, Path::identify(".cpp", false)); //ASSERT_EQUALS(Standards::Language::None, Path::identify(".h", false)); //ASSERT_EQUALS(Standards::Language::None, Path::identify(".hpp", false)); // C ASSERT_EQUALS(Standards::Language::C, Path::identify("index.c", false)); ASSERT_EQUALS(Standards::Language::C, Path::identify("index.cl", false)); ASSERT_EQUALS(Standards::Language::C, Path::identify("C:\\foo\\index.c", false)); ASSERT_EQUALS(Standards::Language::C, Path::identify("/mnt/c/foo/index.c", false)); // In unix .C is considered C++ #ifdef _WIN32 ASSERT_EQUALS(Standards::Language::C, Path::identify("C:\\foo\\index.C", false)); #endif lang = Path::identify("index.c", false, &header); ASSERT_EQUALS(Standards::Language::C, lang); ASSERT_EQUALS(false, header); // C++ ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.cpp", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.cxx", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.cc", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.c++", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.tpp", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.txx", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.ipp", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.ixx", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("C:\\foo\\index.cpp", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("C:\\foo\\index.Cpp", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("/mnt/c/foo/index.cpp", false)); ASSERT_EQUALS(Standards::Language::CPP, Path::identify("/mnt/c/foo/index.Cpp", false)); // TODO: check for case-insenstive filesystem instead // In unix .C is considered C++ #if !defined(_WIN32) && !(defined(__APPLE__) && defined(__MACH__)) ASSERT_EQUALS(Standards::Language::CPP, Path::identify("index.C", false)); #else ASSERT_EQUALS(Standards::Language::C, Path::identify("index.C", false)); #endif lang = Path::identify("index.cpp", false, &header); ASSERT_EQUALS(Standards::Language::CPP, lang); ASSERT_EQUALS(false, header); // headers lang = Path::identify("index.h", false, &header); ASSERT_EQUALS(Standards::Language::C, lang); ASSERT_EQUALS(true, header); lang = Path::identify("index.hpp", false, &header); ASSERT_EQUALS(Standards::Language::CPP, lang); ASSERT_EQUALS(true, header); lang = Path::identify("index.hxx", false, &header); ASSERT_EQUALS(Standards::Language::CPP, lang); ASSERT_EQUALS(true, header); lang = Path::identify("index.h++", false, &header); ASSERT_EQUALS(Standards::Language::CPP, lang); ASSERT_EQUALS(true, header); lang = Path::identify("index.hh", false, &header); ASSERT_EQUALS(Standards::Language::CPP, lang); ASSERT_EQUALS(true, header); ASSERT_EQUALS(Standards::Language::None, Path::identify("index.header", false)); ASSERT_EQUALS(Standards::Language::None, Path::identify("index.htm", false)); ASSERT_EQUALS(Standards::Language::None, Path::identify("index.html", false)); } #define identifyWithCppProbeInternal(...) identifyWithCppProbeInternal_(__FILE__, __LINE__, __VA_ARGS__) void identifyWithCppProbeInternal_(const char* file, int line, const std::string& filename, const std::string& marker, Standards::Language std) const { const ScopedFile f(filename, marker); ASSERT_EQUALS_LOC_MSG(std, Path::identify(f.path(), true), filename + "\n" + marker, file, line); } void identifyWithCppProbe() const { const std::list markers_cpp = { "// -*- C++ -*-", "// -*-C++-*-", "// -*- Mode: C++; -*-", "// -*-Mode: C++;-*-", "// -*- Mode:C++; -*-", "// -*-Mode:C++;-*-", "// -*- Mode: C++ -*-", "// -*-Mode: C++-*-", "// -*- Mode:C++ -*-", "// -*-Mode:C++-*-", "// -*- Mode: C++; c-basic-offset: 8 -*-", "// -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*-", "// -*- c++ -*-", "// -*- mode: c++; -*-", "/* -*- C++ -*- */", "/* -*- C++ -*-", "//-*- C++ -*-", " //-*- C++ -*-", "\t//-*- C++ -*-", "\t //-*- C++ -*-", " \t//-*- C++ -*-", "//-*- C++ -*- ", "//-*- C++ -*- \n", "// -----*- C++ -*-----", "// comment-*- C++ -*-comment", "// -*- C++ -*-\n", "//-*- C++ -*-\r// comment", "//-*- C++ -*-\n// comment", "//-*- C++ -*-\r\n// comment", "/* -*-C++-*- */", "/*-*-C++-*-*/", " /*-*-C++-*-*/", }; for (const auto& f : { "cppprobe.h", "cppprobe" }) { for (const auto& m : markers_cpp) { identifyWithCppProbeInternal(f, m, Standards::Language::CPP); } } const std::list markers_c = { "-*- C++ -*-", // needs to be in comment "// -*- C++", // no end marker "// -*- C++ --*-", // incorrect end marker "// -*- C++/-*-", // unexpected character "// comment\n// -*- C++ -*-", // not on the first line "// comment\r// -*- C++ -*-", // not on the first line "// comment\r\n// -*- C++ -*-", // not on the first line "// -*- C -*-", "// -*- Mode: C; -*-", "// -*- f90 -*-", "// -*- fortran -*-", "// -*- c-basic-offset: 2 -*-", "// -*- c-basic-offset:4; indent-tabs-mode:nil -*-", "// ", // no marker "// -*-", // incomplete marker "/*", // no marker "/**/", // no marker "/*\n*/", // no marker "/* */", // no marker "/* \n*/", // no marker "/* -*-", // incomplete marker "/* \n-*-", // incomplete marker "/* \n-*- C++ -*-", // not on the first line "/* \n-*- C++ -*- */" // not on the first line }; for (const auto& m : markers_c) { identifyWithCppProbeInternal("cppprobe.h", m, Standards::Language::C); } for (const auto& m : markers_c) { identifyWithCppProbeInternal("cppprobe", m, Standards::Language::None); } } void is_header() const { ASSERT(Path::isHeader("index.h")); ASSERT(Path::isHeader("index.hpp")); ASSERT(Path::isHeader("index.hxx")); ASSERT(Path::isHeader("index.h++")); ASSERT(Path::isHeader("index.hh")); ASSERT(Path::isHeader("index.c")==false); ASSERT(Path::isHeader("index.cpp")==false); ASSERT(Path::isHeader("index.header")==false); ASSERT(Path::isHeader("index.htm")==false); ASSERT(Path::isHeader("index.html")==false); } void simplifyPath() const { ASSERT_EQUALS("file.cpp", Path::simplifyPath("file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath("../file.cpp")); ASSERT_EQUALS("file.cpp", Path::simplifyPath("test/../file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath("../test/../file.cpp")); ASSERT_EQUALS("file.cpp", Path::simplifyPath("./file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath("./../file.cpp")); ASSERT_EQUALS("file.cpp", Path::simplifyPath("./test/../file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath("./../test/../file.cpp")); ASSERT_EQUALS("test/", Path::simplifyPath("test/")); ASSERT_EQUALS("../test/", Path::simplifyPath("../test/")); ASSERT_EQUALS("../", Path::simplifyPath("../test/..")); ASSERT_EQUALS("../", Path::simplifyPath("../test/../")); ASSERT_EQUALS("/home/file.cpp", Path::simplifyPath("/home/test/../file.cpp")); ASSERT_EQUALS("/file.cpp", Path::simplifyPath("/home/../test/../file.cpp")); ASSERT_EQUALS("C:/home/file.cpp", Path::simplifyPath("C:/home/test/../file.cpp")); ASSERT_EQUALS("C:/file.cpp", Path::simplifyPath("C:/home/../test/../file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath("..\\file.cpp")); ASSERT_EQUALS("file.cpp", Path::simplifyPath("test\\..\\file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath("..\\test\\..\\file.cpp")); ASSERT_EQUALS("file.cpp", Path::simplifyPath(".\\file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath(".\\..\\file.cpp")); ASSERT_EQUALS("file.cpp", Path::simplifyPath(".\\test\\..\\file.cpp")); ASSERT_EQUALS("../file.cpp", Path::simplifyPath(".\\..\\test\\..\\file.cpp")); ASSERT_EQUALS("test/", Path::simplifyPath("test\\")); ASSERT_EQUALS("../test/", Path::simplifyPath("..\\test\\")); ASSERT_EQUALS("../", Path::simplifyPath("..\\test\\..")); ASSERT_EQUALS("../", Path::simplifyPath("..\\test\\..\\")); ASSERT_EQUALS("C:/home/file.cpp", Path::simplifyPath("C:\\home\\test\\..\\file.cpp")); ASSERT_EQUALS("C:/file.cpp", Path::simplifyPath("C:\\home\\..\\test\\..\\file.cpp")); ASSERT_EQUALS("//home/file.cpp", Path::simplifyPath("\\\\home\\test\\..\\file.cpp")); ASSERT_EQUALS("//file.cpp", Path::simplifyPath("\\\\home\\..\\test\\..\\file.cpp")); } void getAbsolutePath() const { const std::string cwd = Path::getCurrentPath(); ScopedFile file("testabspath.txt", ""); std::string expected = Path::toNativeSeparators(Path::join(cwd, "testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath("testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath("./testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath(".\\testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath("test/../testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath("test\\..\\testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath("./test/../testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath(".\\test\\../testabspath.txt")); ASSERT_EQUALS(expected, Path::getAbsoluteFilePath(Path::join(cwd, "testabspath.txt"))); std::string cwd_up = Path::getPathFromFilename(cwd); cwd_up.pop_back(); // remove trailing slash ASSERT_EQUALS(cwd_up, Path::getAbsoluteFilePath(Path::join(cwd, ".."))); ASSERT_EQUALS(cwd_up, Path::getAbsoluteFilePath(Path::join(cwd, "../"))); ASSERT_EQUALS(cwd_up, Path::getAbsoluteFilePath(Path::join(cwd, "..\\"))); ASSERT_EQUALS(cwd_up, Path::getAbsoluteFilePath(Path::join(cwd, "./../"))); ASSERT_EQUALS(cwd_up, Path::getAbsoluteFilePath(Path::join(cwd, ".\\..\\"))); ASSERT_EQUALS(cwd, Path::getAbsoluteFilePath(".")); #ifndef _WIN32 TODO_ASSERT_EQUALS(cwd, "", Path::getAbsoluteFilePath("./")); TODO_ASSERT_EQUALS(cwd, "", Path::getAbsoluteFilePath(".\\")); #else ASSERT_EQUALS(cwd, Path::getAbsoluteFilePath("./")); ASSERT_EQUALS(cwd, Path::getAbsoluteFilePath(".\\")); #endif ASSERT_EQUALS("", Path::getAbsoluteFilePath("")); #ifndef _WIN32 // the underlying realpath() call only returns something if the path actually exists ASSERT_THROW_EQUALS(Path::getAbsoluteFilePath("testabspath2.txt"), std::runtime_error, "path 'testabspath2.txt' does not exist"); #else ASSERT_EQUALS(Path::toNativeSeparators(Path::join(cwd, "testabspath2.txt")), Path::getAbsoluteFilePath("testabspath2.txt")); #endif #ifdef _WIN32 // determine an existing drive letter std::string drive = Path::getCurrentPath().substr(0, 2); ASSERT_EQUALS(drive + "", Path::getAbsoluteFilePath(drive + "\\")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "\\path")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "\\path\\")); ASSERT_EQUALS(drive + "\\path\\files.txt", Path::getAbsoluteFilePath(drive + "\\path\\files.txt")); ASSERT_EQUALS(drive + "", Path::getAbsoluteFilePath(drive + "//")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "//path")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "//path/")); ASSERT_EQUALS(drive + "\\path\\files.txt", Path::getAbsoluteFilePath(drive + "//path/files.txt")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "\\\\path")); ASSERT_EQUALS(drive + "", Path::getAbsoluteFilePath(drive + "/")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "/path")); drive[0] = static_cast(toupper(drive[0])); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "\\path")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "/path")); drive[0] = static_cast(tolower(drive[0])); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "\\path")); ASSERT_EQUALS(drive + "\\path", Path::getAbsoluteFilePath(drive + "/path")); ASSERT_EQUALS("1:\\path\\files.txt", Path::getAbsoluteFilePath("1:\\path\\files.txt")); // treated as valid drive ASSERT_EQUALS( Path::toNativeSeparators(Path::join(Path::getCurrentPath(), "CC:\\path\\files.txt")), Path::getAbsoluteFilePath("CC:\\path\\files.txt")); // treated as filename ASSERT_EQUALS("1:\\path\\files.txt", Path::getAbsoluteFilePath("1:/path/files.txt")); // treated as valid drive ASSERT_EQUALS( Path::toNativeSeparators(Path::join(Path::getCurrentPath(), "CC:\\path\\files.txt")), Path::getAbsoluteFilePath("CC:/path/files.txt")); // treated as filename #endif #ifndef _WIN32 ASSERT_THROW_EQUALS(Path::getAbsoluteFilePath("C:\\path\\files.txt"), std::runtime_error, "path 'C:\\path\\files.txt' does not exist"); #endif // TODO: test UNC paths // TODO: test with symlinks } void exists() const { ScopedFile file("testpath.txt", "", "testpath"); ScopedFile file2("testpath2.txt", ""); ASSERT_EQUALS(true, Path::exists("testpath")); ASSERT_EQUALS(true, Path::exists("testpath2.txt")); ASSERT_EQUALS(false, Path::exists("testpath2")); bool b = false; ASSERT_EQUALS(true, Path::exists("testpath", &b)); ASSERT_EQUALS(true, b); ASSERT_EQUALS(true, Path::exists("testpath/testpath.txt", &b)); ASSERT_EQUALS(false, b); ASSERT_EQUALS(true, Path::exists("testpath2.txt", &b)); ASSERT_EQUALS(false, b); ASSERT_EQUALS(false, Path::exists("testpath2", &b)); ASSERT_EQUALS(false, b); ASSERT_EQUALS(false, Path::exists("testpath/testpath2.txt", &b)); ASSERT_EQUALS(false, b); ASSERT_EQUALS(false, Path::exists("testpath.txt", &b)); ASSERT_EQUALS(false, b); } void fromNativeSeparators() const { ASSERT_EQUALS("lib/file.c", Path::fromNativeSeparators("lib/file.c")); ASSERT_EQUALS("lib//file.c", Path::fromNativeSeparators("lib//file.c")); ASSERT_EQUALS("/lib/file.c", Path::fromNativeSeparators("/lib/file.c")); ASSERT_EQUALS("//lib/file.c", Path::fromNativeSeparators("//lib/file.c")); ASSERT_EQUALS("./lib/file.c", Path::fromNativeSeparators("./lib/file.c")); ASSERT_EQUALS("lib/file.c", Path::fromNativeSeparators("lib\\file.c")); ASSERT_EQUALS("lib//file.c", Path::fromNativeSeparators("lib\\\\file.c")); ASSERT_EQUALS("/lib/file.c", Path::fromNativeSeparators("\\lib\\file.c")); ASSERT_EQUALS("//lib/file.c", Path::fromNativeSeparators("\\\\lib\\file.c")); ASSERT_EQUALS("./lib/file.c", Path::fromNativeSeparators(".\\lib\\file.c")); } }; REGISTER_TEST(TestPath)
X Tutup