googletest-filepath-test.cc 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671
  1. // Copyright 2008, Google Inc.
  2. // All rights reserved.
  3. //
  4. // Redistribution and use in source and binary forms, with or without
  5. // modification, are permitted provided that the following conditions are
  6. // met:
  7. //
  8. // * Redistributions of source code must retain the above copyright
  9. // notice, this list of conditions and the following disclaimer.
  10. // * Redistributions in binary form must reproduce the above
  11. // copyright notice, this list of conditions and the following disclaimer
  12. // in the documentation and/or other materials provided with the
  13. // distribution.
  14. // * Neither the name of Google Inc. nor the names of its
  15. // contributors may be used to endorse or promote products derived from
  16. // this software without specific prior written permission.
  17. //
  18. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  19. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  20. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  21. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  22. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  23. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  24. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  25. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  26. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  27. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  28. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  29. //
  30. // Google Test filepath utilities
  31. //
  32. // This file tests classes and functions used internally by
  33. // Google Test. They are subject to change without notice.
  34. //
  35. // This file is #included from gtest-internal.h.
  36. // Do not #include this file anywhere else!
  37. #include <string>
  38. #include "gtest/gtest.h"
  39. #include "gtest/internal/gtest-filepath.h"
  40. #include "src/gtest-internal-inl.h"
  41. #if GTEST_OS_WINDOWS_MOBILE
  42. #include <windows.h> // NOLINT
  43. #elif GTEST_OS_WINDOWS
  44. #include <direct.h> // NOLINT
  45. #endif // GTEST_OS_WINDOWS_MOBILE
  46. namespace testing {
  47. namespace internal {
  48. namespace {
  49. #if GTEST_OS_WINDOWS_MOBILE
  50. // Windows CE doesn't have the remove C function.
  51. int remove(const char* path) {
  52. LPCWSTR wpath = String::AnsiToUtf16(path);
  53. int ret = DeleteFile(wpath) ? 0 : -1;
  54. delete[] wpath;
  55. return ret;
  56. }
  57. // Windows CE doesn't have the _rmdir C function.
  58. int _rmdir(const char* path) {
  59. FilePath filepath(path);
  60. LPCWSTR wpath =
  61. String::AnsiToUtf16(filepath.RemoveTrailingPathSeparator().c_str());
  62. int ret = RemoveDirectory(wpath) ? 0 : -1;
  63. delete[] wpath;
  64. return ret;
  65. }
  66. #else
  67. TEST(GetCurrentDirTest, ReturnsCurrentDir) {
  68. const FilePath original_dir = FilePath::GetCurrentDir();
  69. EXPECT_FALSE(original_dir.IsEmpty());
  70. posix::ChDir(GTEST_PATH_SEP_);
  71. const FilePath cwd = FilePath::GetCurrentDir();
  72. posix::ChDir(original_dir.c_str());
  73. #if GTEST_OS_WINDOWS || GTEST_OS_OS2
  74. // Skips the ":".
  75. const char* const cwd_without_drive = strchr(cwd.c_str(), ':');
  76. ASSERT_TRUE(cwd_without_drive != NULL);
  77. EXPECT_STREQ(GTEST_PATH_SEP_, cwd_without_drive + 1);
  78. #else
  79. EXPECT_EQ(GTEST_PATH_SEP_, cwd.string());
  80. #endif
  81. }
  82. #endif // GTEST_OS_WINDOWS_MOBILE
  83. TEST(IsEmptyTest, ReturnsTrueForEmptyPath) {
  84. EXPECT_TRUE(FilePath("").IsEmpty());
  85. }
  86. TEST(IsEmptyTest, ReturnsFalseForNonEmptyPath) {
  87. EXPECT_FALSE(FilePath("a").IsEmpty());
  88. EXPECT_FALSE(FilePath(".").IsEmpty());
  89. EXPECT_FALSE(FilePath("a/b").IsEmpty());
  90. EXPECT_FALSE(FilePath("a\\b\\").IsEmpty());
  91. }
  92. // RemoveDirectoryName "" -> ""
  93. TEST(RemoveDirectoryNameTest, WhenEmptyName) {
  94. EXPECT_EQ("", FilePath("").RemoveDirectoryName().string());
  95. }
  96. // RemoveDirectoryName "afile" -> "afile"
  97. TEST(RemoveDirectoryNameTest, ButNoDirectory) {
  98. EXPECT_EQ("afile", FilePath("afile").RemoveDirectoryName().string());
  99. }
  100. // RemoveDirectoryName "/afile" -> "afile"
  101. TEST(RemoveDirectoryNameTest, RootFileShouldGiveFileName) {
  102. EXPECT_EQ("afile",
  103. FilePath(GTEST_PATH_SEP_ "afile").RemoveDirectoryName().string());
  104. }
  105. // RemoveDirectoryName "adir/" -> ""
  106. TEST(RemoveDirectoryNameTest, WhereThereIsNoFileName) {
  107. EXPECT_EQ("",
  108. FilePath("adir" GTEST_PATH_SEP_).RemoveDirectoryName().string());
  109. }
  110. // RemoveDirectoryName "adir/afile" -> "afile"
  111. TEST(RemoveDirectoryNameTest, ShouldGiveFileName) {
  112. EXPECT_EQ(
  113. "afile",
  114. FilePath("adir" GTEST_PATH_SEP_ "afile").RemoveDirectoryName().string());
  115. }
  116. // RemoveDirectoryName "adir/subdir/afile" -> "afile"
  117. TEST(RemoveDirectoryNameTest, ShouldAlsoGiveFileName) {
  118. EXPECT_EQ("afile",
  119. FilePath("adir" GTEST_PATH_SEP_ "subdir" GTEST_PATH_SEP_ "afile")
  120. .RemoveDirectoryName()
  121. .string());
  122. }
  123. #if GTEST_HAS_ALT_PATH_SEP_
  124. // Tests that RemoveDirectoryName() works with the alternate separator
  125. // on Windows.
  126. // RemoveDirectoryName("/afile") -> "afile"
  127. TEST(RemoveDirectoryNameTest, RootFileShouldGiveFileNameForAlternateSeparator) {
  128. EXPECT_EQ("afile", FilePath("/afile").RemoveDirectoryName().string());
  129. }
  130. // RemoveDirectoryName("adir/") -> ""
  131. TEST(RemoveDirectoryNameTest, WhereThereIsNoFileNameForAlternateSeparator) {
  132. EXPECT_EQ("", FilePath("adir/").RemoveDirectoryName().string());
  133. }
  134. // RemoveDirectoryName("adir/afile") -> "afile"
  135. TEST(RemoveDirectoryNameTest, ShouldGiveFileNameForAlternateSeparator) {
  136. EXPECT_EQ("afile", FilePath("adir/afile").RemoveDirectoryName().string());
  137. }
  138. // RemoveDirectoryName("adir/subdir/afile") -> "afile"
  139. TEST(RemoveDirectoryNameTest, ShouldAlsoGiveFileNameForAlternateSeparator) {
  140. EXPECT_EQ("afile",
  141. FilePath("adir/subdir/afile").RemoveDirectoryName().string());
  142. }
  143. #endif
  144. // RemoveFileName "" -> "./"
  145. TEST(RemoveFileNameTest, EmptyName) {
  146. #if GTEST_OS_WINDOWS_MOBILE
  147. // On Windows CE, we use the root as the current directory.
  148. EXPECT_EQ(GTEST_PATH_SEP_, FilePath("").RemoveFileName().string());
  149. #else
  150. EXPECT_EQ("." GTEST_PATH_SEP_, FilePath("").RemoveFileName().string());
  151. #endif
  152. }
  153. // RemoveFileName "adir/" -> "adir/"
  154. TEST(RemoveFileNameTest, ButNoFile) {
  155. EXPECT_EQ("adir" GTEST_PATH_SEP_,
  156. FilePath("adir" GTEST_PATH_SEP_).RemoveFileName().string());
  157. }
  158. // RemoveFileName "adir/afile" -> "adir/"
  159. TEST(RemoveFileNameTest, GivesDirName) {
  160. EXPECT_EQ("adir" GTEST_PATH_SEP_,
  161. FilePath("adir" GTEST_PATH_SEP_ "afile").RemoveFileName().string());
  162. }
  163. // RemoveFileName "adir/subdir/afile" -> "adir/subdir/"
  164. TEST(RemoveFileNameTest, GivesDirAndSubDirName) {
  165. EXPECT_EQ("adir" GTEST_PATH_SEP_ "subdir" GTEST_PATH_SEP_,
  166. FilePath("adir" GTEST_PATH_SEP_ "subdir" GTEST_PATH_SEP_ "afile")
  167. .RemoveFileName()
  168. .string());
  169. }
  170. // RemoveFileName "/afile" -> "/"
  171. TEST(RemoveFileNameTest, GivesRootDir) {
  172. EXPECT_EQ(GTEST_PATH_SEP_,
  173. FilePath(GTEST_PATH_SEP_ "afile").RemoveFileName().string());
  174. }
  175. #if GTEST_HAS_ALT_PATH_SEP_
  176. // Tests that RemoveFileName() works with the alternate separator on
  177. // Windows.
  178. // RemoveFileName("adir/") -> "adir/"
  179. TEST(RemoveFileNameTest, ButNoFileForAlternateSeparator) {
  180. EXPECT_EQ("adir" GTEST_PATH_SEP_,
  181. FilePath("adir/").RemoveFileName().string());
  182. }
  183. // RemoveFileName("adir/afile") -> "adir/"
  184. TEST(RemoveFileNameTest, GivesDirNameForAlternateSeparator) {
  185. EXPECT_EQ("adir" GTEST_PATH_SEP_,
  186. FilePath("adir/afile").RemoveFileName().string());
  187. }
  188. // RemoveFileName("adir/subdir/afile") -> "adir/subdir/"
  189. TEST(RemoveFileNameTest, GivesDirAndSubDirNameForAlternateSeparator) {
  190. EXPECT_EQ("adir" GTEST_PATH_SEP_ "subdir" GTEST_PATH_SEP_,
  191. FilePath("adir/subdir/afile").RemoveFileName().string());
  192. }
  193. // RemoveFileName("/afile") -> "\"
  194. TEST(RemoveFileNameTest, GivesRootDirForAlternateSeparator) {
  195. EXPECT_EQ(GTEST_PATH_SEP_, FilePath("/afile").RemoveFileName().string());
  196. }
  197. #endif
  198. TEST(MakeFileNameTest, GenerateWhenNumberIsZero) {
  199. FilePath actual =
  200. FilePath::MakeFileName(FilePath("foo"), FilePath("bar"), 0, "xml");
  201. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar.xml", actual.string());
  202. }
  203. TEST(MakeFileNameTest, GenerateFileNameNumberGtZero) {
  204. FilePath actual =
  205. FilePath::MakeFileName(FilePath("foo"), FilePath("bar"), 12, "xml");
  206. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar_12.xml", actual.string());
  207. }
  208. TEST(MakeFileNameTest, GenerateFileNameWithSlashNumberIsZero) {
  209. FilePath actual = FilePath::MakeFileName(FilePath("foo" GTEST_PATH_SEP_),
  210. FilePath("bar"), 0, "xml");
  211. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar.xml", actual.string());
  212. }
  213. TEST(MakeFileNameTest, GenerateFileNameWithSlashNumberGtZero) {
  214. FilePath actual = FilePath::MakeFileName(FilePath("foo" GTEST_PATH_SEP_),
  215. FilePath("bar"), 12, "xml");
  216. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar_12.xml", actual.string());
  217. }
  218. TEST(MakeFileNameTest, GenerateWhenNumberIsZeroAndDirIsEmpty) {
  219. FilePath actual =
  220. FilePath::MakeFileName(FilePath(""), FilePath("bar"), 0, "xml");
  221. EXPECT_EQ("bar.xml", actual.string());
  222. }
  223. TEST(MakeFileNameTest, GenerateWhenNumberIsNotZeroAndDirIsEmpty) {
  224. FilePath actual =
  225. FilePath::MakeFileName(FilePath(""), FilePath("bar"), 14, "xml");
  226. EXPECT_EQ("bar_14.xml", actual.string());
  227. }
  228. TEST(ConcatPathsTest, WorksWhenDirDoesNotEndWithPathSep) {
  229. FilePath actual = FilePath::ConcatPaths(FilePath("foo"), FilePath("bar.xml"));
  230. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar.xml", actual.string());
  231. }
  232. TEST(ConcatPathsTest, WorksWhenPath1EndsWithPathSep) {
  233. FilePath actual = FilePath::ConcatPaths(FilePath("foo" GTEST_PATH_SEP_),
  234. FilePath("bar.xml"));
  235. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar.xml", actual.string());
  236. }
  237. TEST(ConcatPathsTest, Path1BeingEmpty) {
  238. FilePath actual = FilePath::ConcatPaths(FilePath(""), FilePath("bar.xml"));
  239. EXPECT_EQ("bar.xml", actual.string());
  240. }
  241. TEST(ConcatPathsTest, Path2BeingEmpty) {
  242. FilePath actual = FilePath::ConcatPaths(FilePath("foo"), FilePath(""));
  243. EXPECT_EQ("foo" GTEST_PATH_SEP_, actual.string());
  244. }
  245. TEST(ConcatPathsTest, BothPathBeingEmpty) {
  246. FilePath actual = FilePath::ConcatPaths(FilePath(""), FilePath(""));
  247. EXPECT_EQ("", actual.string());
  248. }
  249. TEST(ConcatPathsTest, Path1ContainsPathSep) {
  250. FilePath actual = FilePath::ConcatPaths(FilePath("foo" GTEST_PATH_SEP_ "bar"),
  251. FilePath("foobar.xml"));
  252. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar" GTEST_PATH_SEP_ "foobar.xml",
  253. actual.string());
  254. }
  255. TEST(ConcatPathsTest, Path2ContainsPathSep) {
  256. FilePath actual =
  257. FilePath::ConcatPaths(FilePath("foo" GTEST_PATH_SEP_),
  258. FilePath("bar" GTEST_PATH_SEP_ "bar.xml"));
  259. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar" GTEST_PATH_SEP_ "bar.xml",
  260. actual.string());
  261. }
  262. TEST(ConcatPathsTest, Path2EndsWithPathSep) {
  263. FilePath actual =
  264. FilePath::ConcatPaths(FilePath("foo"), FilePath("bar" GTEST_PATH_SEP_));
  265. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar" GTEST_PATH_SEP_, actual.string());
  266. }
  267. // RemoveTrailingPathSeparator "" -> ""
  268. TEST(RemoveTrailingPathSeparatorTest, EmptyString) {
  269. EXPECT_EQ("", FilePath("").RemoveTrailingPathSeparator().string());
  270. }
  271. // RemoveTrailingPathSeparator "foo" -> "foo"
  272. TEST(RemoveTrailingPathSeparatorTest, FileNoSlashString) {
  273. EXPECT_EQ("foo", FilePath("foo").RemoveTrailingPathSeparator().string());
  274. }
  275. // RemoveTrailingPathSeparator "foo/" -> "foo"
  276. TEST(RemoveTrailingPathSeparatorTest, ShouldRemoveTrailingSeparator) {
  277. EXPECT_EQ(
  278. "foo",
  279. FilePath("foo" GTEST_PATH_SEP_).RemoveTrailingPathSeparator().string());
  280. #if GTEST_HAS_ALT_PATH_SEP_
  281. EXPECT_EQ("foo", FilePath("foo/").RemoveTrailingPathSeparator().string());
  282. #endif
  283. }
  284. // RemoveTrailingPathSeparator "foo/bar/" -> "foo/bar/"
  285. TEST(RemoveTrailingPathSeparatorTest, ShouldRemoveLastSeparator) {
  286. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar",
  287. FilePath("foo" GTEST_PATH_SEP_ "bar" GTEST_PATH_SEP_)
  288. .RemoveTrailingPathSeparator()
  289. .string());
  290. }
  291. // RemoveTrailingPathSeparator "foo/bar" -> "foo/bar"
  292. TEST(RemoveTrailingPathSeparatorTest, ShouldReturnUnmodified) {
  293. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar", FilePath("foo" GTEST_PATH_SEP_ "bar")
  294. .RemoveTrailingPathSeparator()
  295. .string());
  296. }
  297. TEST(DirectoryTest, RootDirectoryExists) {
  298. #if GTEST_OS_WINDOWS // We are on Windows.
  299. char current_drive[_MAX_PATH]; // NOLINT
  300. current_drive[0] = static_cast<char>(_getdrive() + 'A' - 1);
  301. current_drive[1] = ':';
  302. current_drive[2] = '\\';
  303. current_drive[3] = '\0';
  304. EXPECT_TRUE(FilePath(current_drive).DirectoryExists());
  305. #else
  306. EXPECT_TRUE(FilePath("/").DirectoryExists());
  307. #endif // GTEST_OS_WINDOWS
  308. }
  309. #if GTEST_OS_WINDOWS
  310. TEST(DirectoryTest, RootOfWrongDriveDoesNotExists) {
  311. const int saved_drive_ = _getdrive();
  312. // Find a drive that doesn't exist. Start with 'Z' to avoid common ones.
  313. for (char drive = 'Z'; drive >= 'A'; drive--)
  314. if (_chdrive(drive - 'A' + 1) == -1) {
  315. char non_drive[_MAX_PATH]; // NOLINT
  316. non_drive[0] = drive;
  317. non_drive[1] = ':';
  318. non_drive[2] = '\\';
  319. non_drive[3] = '\0';
  320. EXPECT_FALSE(FilePath(non_drive).DirectoryExists());
  321. break;
  322. }
  323. _chdrive(saved_drive_);
  324. }
  325. #endif // GTEST_OS_WINDOWS
  326. #if !GTEST_OS_WINDOWS_MOBILE
  327. // Windows CE _does_ consider an empty directory to exist.
  328. TEST(DirectoryTest, EmptyPathDirectoryDoesNotExist) {
  329. EXPECT_FALSE(FilePath("").DirectoryExists());
  330. }
  331. #endif // !GTEST_OS_WINDOWS_MOBILE
  332. TEST(DirectoryTest, CurrentDirectoryExists) {
  333. #if GTEST_OS_WINDOWS // We are on Windows.
  334. #ifndef _WIN32_CE // Windows CE doesn't have a current directory.
  335. EXPECT_TRUE(FilePath(".").DirectoryExists());
  336. EXPECT_TRUE(FilePath(".\\").DirectoryExists());
  337. #endif // _WIN32_CE
  338. #else
  339. EXPECT_TRUE(FilePath(".").DirectoryExists());
  340. EXPECT_TRUE(FilePath("./").DirectoryExists());
  341. #endif // GTEST_OS_WINDOWS
  342. }
  343. // "foo/bar" == foo//bar" == "foo///bar"
  344. TEST(NormalizeTest, MultipleConsecutiveSeparatorsInMidstring) {
  345. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar",
  346. FilePath("foo" GTEST_PATH_SEP_ "bar").string());
  347. EXPECT_EQ("foo" GTEST_PATH_SEP_ "bar",
  348. FilePath("foo" GTEST_PATH_SEP_ GTEST_PATH_SEP_ "bar").string());
  349. EXPECT_EQ(
  350. "foo" GTEST_PATH_SEP_ "bar",
  351. FilePath("foo" GTEST_PATH_SEP_ GTEST_PATH_SEP_ GTEST_PATH_SEP_ "bar")
  352. .string());
  353. }
  354. // "/bar" == //bar" == "///bar"
  355. TEST(NormalizeTest, MultipleConsecutiveSeparatorsAtStringStart) {
  356. EXPECT_EQ(GTEST_PATH_SEP_ "bar", FilePath(GTEST_PATH_SEP_ "bar").string());
  357. #if GTEST_OS_WINDOWS
  358. EXPECT_EQ(GTEST_PATH_SEP_ GTEST_PATH_SEP_ "bar",
  359. FilePath(GTEST_PATH_SEP_ GTEST_PATH_SEP_ "bar").string());
  360. #else
  361. EXPECT_EQ(GTEST_PATH_SEP_ "bar",
  362. FilePath(GTEST_PATH_SEP_ GTEST_PATH_SEP_ "bar").string());
  363. #endif
  364. EXPECT_EQ(
  365. GTEST_PATH_SEP_ "bar",
  366. FilePath(GTEST_PATH_SEP_ GTEST_PATH_SEP_ GTEST_PATH_SEP_ "bar").string());
  367. }
  368. // "foo/" == foo//" == "foo///"
  369. TEST(NormalizeTest, MultipleConsecutiveSeparatorsAtStringEnd) {
  370. EXPECT_EQ("foo" GTEST_PATH_SEP_, FilePath("foo" GTEST_PATH_SEP_).string());
  371. EXPECT_EQ("foo" GTEST_PATH_SEP_,
  372. FilePath("foo" GTEST_PATH_SEP_ GTEST_PATH_SEP_).string());
  373. EXPECT_EQ(
  374. "foo" GTEST_PATH_SEP_,
  375. FilePath("foo" GTEST_PATH_SEP_ GTEST_PATH_SEP_ GTEST_PATH_SEP_).string());
  376. }
  377. #if GTEST_HAS_ALT_PATH_SEP_
  378. // Tests that separators at the end of the string are normalized
  379. // regardless of their combination (e.g. "foo\" =="foo/\" ==
  380. // "foo\\/").
  381. TEST(NormalizeTest, MixAlternateSeparatorAtStringEnd) {
  382. EXPECT_EQ("foo" GTEST_PATH_SEP_, FilePath("foo/").string());
  383. EXPECT_EQ("foo" GTEST_PATH_SEP_,
  384. FilePath("foo" GTEST_PATH_SEP_ "/").string());
  385. EXPECT_EQ("foo" GTEST_PATH_SEP_, FilePath("foo//" GTEST_PATH_SEP_).string());
  386. }
  387. #endif
  388. TEST(AssignmentOperatorTest, DefaultAssignedToNonDefault) {
  389. FilePath default_path;
  390. FilePath non_default_path("path");
  391. non_default_path = default_path;
  392. EXPECT_EQ("", non_default_path.string());
  393. EXPECT_EQ("", default_path.string()); // RHS var is unchanged.
  394. }
  395. TEST(AssignmentOperatorTest, NonDefaultAssignedToDefault) {
  396. FilePath non_default_path("path");
  397. FilePath default_path;
  398. default_path = non_default_path;
  399. EXPECT_EQ("path", default_path.string());
  400. EXPECT_EQ("path", non_default_path.string()); // RHS var is unchanged.
  401. }
  402. TEST(AssignmentOperatorTest, ConstAssignedToNonConst) {
  403. const FilePath const_default_path("const_path");
  404. FilePath non_default_path("path");
  405. non_default_path = const_default_path;
  406. EXPECT_EQ("const_path", non_default_path.string());
  407. }
  408. class DirectoryCreationTest : public Test {
  409. protected:
  410. void SetUp() override {
  411. testdata_path_.Set(
  412. FilePath(TempDir() + GetCurrentExecutableName().string() +
  413. "_directory_creation" GTEST_PATH_SEP_ "test" GTEST_PATH_SEP_));
  414. testdata_file_.Set(testdata_path_.RemoveTrailingPathSeparator());
  415. unique_file0_.Set(
  416. FilePath::MakeFileName(testdata_path_, FilePath("unique"), 0, "txt"));
  417. unique_file1_.Set(
  418. FilePath::MakeFileName(testdata_path_, FilePath("unique"), 1, "txt"));
  419. remove(testdata_file_.c_str());
  420. remove(unique_file0_.c_str());
  421. remove(unique_file1_.c_str());
  422. posix::RmDir(testdata_path_.c_str());
  423. }
  424. void TearDown() override {
  425. remove(testdata_file_.c_str());
  426. remove(unique_file0_.c_str());
  427. remove(unique_file1_.c_str());
  428. posix::RmDir(testdata_path_.c_str());
  429. }
  430. void CreateTextFile(const char* filename) {
  431. FILE* f = posix::FOpen(filename, "w");
  432. fprintf(f, "text\n");
  433. fclose(f);
  434. }
  435. // Strings representing a directory and a file, with identical paths
  436. // except for the trailing separator character that distinquishes
  437. // a directory named 'test' from a file named 'test'. Example names:
  438. FilePath testdata_path_; // "/tmp/directory_creation/test/"
  439. FilePath testdata_file_; // "/tmp/directory_creation/test"
  440. FilePath unique_file0_; // "/tmp/directory_creation/test/unique.txt"
  441. FilePath unique_file1_; // "/tmp/directory_creation/test/unique_1.txt"
  442. };
  443. TEST_F(DirectoryCreationTest, CreateDirectoriesRecursively) {
  444. EXPECT_FALSE(testdata_path_.DirectoryExists()) << testdata_path_.string();
  445. EXPECT_TRUE(testdata_path_.CreateDirectoriesRecursively());
  446. EXPECT_TRUE(testdata_path_.DirectoryExists());
  447. }
  448. TEST_F(DirectoryCreationTest, CreateDirectoriesForAlreadyExistingPath) {
  449. EXPECT_FALSE(testdata_path_.DirectoryExists()) << testdata_path_.string();
  450. EXPECT_TRUE(testdata_path_.CreateDirectoriesRecursively());
  451. // Call 'create' again... should still succeed.
  452. EXPECT_TRUE(testdata_path_.CreateDirectoriesRecursively());
  453. }
  454. TEST_F(DirectoryCreationTest, CreateDirectoriesAndUniqueFilename) {
  455. FilePath file_path(FilePath::GenerateUniqueFileName(
  456. testdata_path_, FilePath("unique"), "txt"));
  457. EXPECT_EQ(unique_file0_.string(), file_path.string());
  458. EXPECT_FALSE(file_path.FileOrDirectoryExists()); // file not there
  459. testdata_path_.CreateDirectoriesRecursively();
  460. EXPECT_FALSE(file_path.FileOrDirectoryExists()); // file still not there
  461. CreateTextFile(file_path.c_str());
  462. EXPECT_TRUE(file_path.FileOrDirectoryExists());
  463. FilePath file_path2(FilePath::GenerateUniqueFileName(
  464. testdata_path_, FilePath("unique"), "txt"));
  465. EXPECT_EQ(unique_file1_.string(), file_path2.string());
  466. EXPECT_FALSE(file_path2.FileOrDirectoryExists()); // file not there
  467. CreateTextFile(file_path2.c_str());
  468. EXPECT_TRUE(file_path2.FileOrDirectoryExists());
  469. }
  470. TEST_F(DirectoryCreationTest, CreateDirectoriesFail) {
  471. // force a failure by putting a file where we will try to create a directory.
  472. CreateTextFile(testdata_file_.c_str());
  473. EXPECT_TRUE(testdata_file_.FileOrDirectoryExists());
  474. EXPECT_FALSE(testdata_file_.DirectoryExists());
  475. EXPECT_FALSE(testdata_file_.CreateDirectoriesRecursively());
  476. }
  477. TEST(NoDirectoryCreationTest, CreateNoDirectoriesForDefaultXmlFile) {
  478. const FilePath test_detail_xml("test_detail.xml");
  479. EXPECT_FALSE(test_detail_xml.CreateDirectoriesRecursively());
  480. }
  481. TEST(FilePathTest, DefaultConstructor) {
  482. FilePath fp;
  483. EXPECT_EQ("", fp.string());
  484. }
  485. TEST(FilePathTest, CharAndCopyConstructors) {
  486. const FilePath fp("spicy");
  487. EXPECT_EQ("spicy", fp.string());
  488. const FilePath fp_copy(fp);
  489. EXPECT_EQ("spicy", fp_copy.string());
  490. }
  491. TEST(FilePathTest, StringConstructor) {
  492. const FilePath fp(std::string("cider"));
  493. EXPECT_EQ("cider", fp.string());
  494. }
  495. TEST(FilePathTest, Set) {
  496. const FilePath apple("apple");
  497. FilePath mac("mac");
  498. mac.Set(apple); // Implement Set() since overloading operator= is forbidden.
  499. EXPECT_EQ("apple", mac.string());
  500. EXPECT_EQ("apple", apple.string());
  501. }
  502. TEST(FilePathTest, ToString) {
  503. const FilePath file("drink");
  504. EXPECT_EQ("drink", file.string());
  505. }
  506. TEST(FilePathTest, RemoveExtension) {
  507. EXPECT_EQ("app", FilePath("app.cc").RemoveExtension("cc").string());
  508. EXPECT_EQ("app", FilePath("app.exe").RemoveExtension("exe").string());
  509. EXPECT_EQ("APP", FilePath("APP.EXE").RemoveExtension("exe").string());
  510. }
  511. TEST(FilePathTest, RemoveExtensionWhenThereIsNoExtension) {
  512. EXPECT_EQ("app", FilePath("app").RemoveExtension("exe").string());
  513. }
  514. TEST(FilePathTest, IsDirectory) {
  515. EXPECT_FALSE(FilePath("cola").IsDirectory());
  516. EXPECT_TRUE(FilePath("koala" GTEST_PATH_SEP_).IsDirectory());
  517. #if GTEST_HAS_ALT_PATH_SEP_
  518. EXPECT_TRUE(FilePath("koala/").IsDirectory());
  519. #endif
  520. }
  521. TEST(FilePathTest, IsAbsolutePath) {
  522. EXPECT_FALSE(FilePath("is" GTEST_PATH_SEP_ "relative").IsAbsolutePath());
  523. EXPECT_FALSE(FilePath("").IsAbsolutePath());
  524. #if GTEST_OS_WINDOWS
  525. EXPECT_TRUE(
  526. FilePath("c:\\" GTEST_PATH_SEP_ "is_not" GTEST_PATH_SEP_ "relative")
  527. .IsAbsolutePath());
  528. EXPECT_FALSE(FilePath("c:foo" GTEST_PATH_SEP_ "bar").IsAbsolutePath());
  529. EXPECT_TRUE(
  530. FilePath("c:/" GTEST_PATH_SEP_ "is_not" GTEST_PATH_SEP_ "relative")
  531. .IsAbsolutePath());
  532. EXPECT_TRUE(FilePath("d:/Windows").IsAbsolutePath());
  533. EXPECT_TRUE(FilePath("\\\\Host\\Share").IsAbsolutePath());
  534. EXPECT_TRUE(FilePath("\\\\Host\\Share\\Folder").IsAbsolutePath());
  535. #else
  536. EXPECT_TRUE(FilePath(GTEST_PATH_SEP_ "is_not" GTEST_PATH_SEP_ "relative")
  537. .IsAbsolutePath());
  538. #endif // GTEST_OS_WINDOWS
  539. }
  540. TEST(FilePathTest, IsRootDirectory) {
  541. #if GTEST_OS_WINDOWS
  542. EXPECT_TRUE(FilePath("a:\\").IsRootDirectory());
  543. EXPECT_TRUE(FilePath("Z:/").IsRootDirectory());
  544. EXPECT_TRUE(FilePath("e://").IsRootDirectory());
  545. EXPECT_FALSE(FilePath("").IsRootDirectory());
  546. EXPECT_FALSE(FilePath("b:").IsRootDirectory());
  547. EXPECT_FALSE(FilePath("b:a").IsRootDirectory());
  548. EXPECT_FALSE(FilePath("8:/").IsRootDirectory());
  549. EXPECT_FALSE(FilePath("c|/").IsRootDirectory());
  550. EXPECT_TRUE(FilePath("c:/").IsRootDirectory());
  551. EXPECT_FALSE(FilePath("d:/Windows").IsRootDirectory());
  552. // This is for backward compatibility, since callers (even in this library)
  553. // have assumed IsRootDirectory() implies a trailing directory separator.
  554. EXPECT_FALSE(FilePath("\\\\Host\\Share").IsRootDirectory());
  555. EXPECT_TRUE(FilePath("\\\\Host\\Share\\").IsRootDirectory());
  556. EXPECT_FALSE(FilePath("\\\\Host\\Share\\.").IsRootDirectory());
  557. EXPECT_FALSE(FilePath("\\\\Host\\Share\\C$\\").IsRootDirectory());
  558. #else
  559. EXPECT_TRUE(FilePath("/").IsRootDirectory());
  560. EXPECT_TRUE(FilePath("//").IsRootDirectory());
  561. EXPECT_FALSE(FilePath("").IsRootDirectory());
  562. EXPECT_FALSE(FilePath("\\").IsRootDirectory());
  563. EXPECT_FALSE(FilePath("/x").IsRootDirectory());
  564. #endif
  565. }
  566. } // namespace
  567. } // namespace internal
  568. } // namespace testing