1 //===----------------------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 // REQUIRES: can-create-symlinks
10 // UNSUPPORTED: c++03, c++11, c++14
11 // UNSUPPORTED: no-filesystem
12 // UNSUPPORTED: availability-filesystem-missing
13 
14 // <filesystem>
15 
16 // path weakly_canonical(const path& p);
17 // path weakly_canonical(const path& p, error_code& ec);
18 
19 #include <filesystem>
20 #include <string>
21 
22 #include "assert_macros.h"
23 #include "concat_macros.h"
24 #include "test_macros.h"
25 #include "test_iterators.h"
26 #include "count_new.h"
27 #include "filesystem_test_helper.h"
28 #include "../../class.path/path_helper.h"
29 namespace fs = std::filesystem;
30 
main(int,char **)31 int main(int, char**) {
32   static_test_env static_env;
33 
34   fs::path root = fs::current_path().root_path();
35   // clang-format off
36   struct {
37     fs::path input;
38     fs::path expect;
39   } TestCases[] = {
40       {"", fs::current_path()},
41       {".", fs::current_path()},
42       {"/", root},
43       {"/foo", root / "foo"},
44       {"/.", root},
45       {"/./", root},
46       {"a/b", fs::current_path() / "a/b"},
47       {"a", fs::current_path() / "a"},
48       {"a/b/", fs::current_path() / "a/b/"},
49       {static_env.File, static_env.File},
50       {static_env.Dir, static_env.Dir},
51       {static_env.SymlinkToDir, static_env.Dir},
52       {static_env.SymlinkToDir / "dir2/.", static_env.Dir / "dir2"},
53       // Note: If the trailing separator occurs in a part of the path that exists,
54       // it is omitted. Otherwise it is added to the end of the result.
55       // MS STL and libstdc++ behave similarly.
56       {static_env.SymlinkToDir / "dir2/./", static_env.Dir / "dir2"},
57       {static_env.SymlinkToDir / "dir2/DNE/./", static_env.Dir / "dir2/DNE/"},
58       {static_env.SymlinkToDir / "dir2", static_env.Dir2},
59 #ifdef _WIN32
60       // On Windows, this path is considered to exist (even though it
61       // passes through a nonexistent directory), and thus is returned
62       // without a trailing slash, see the note above.
63       {static_env.SymlinkToDir / "dir2/../dir2/DNE/..", static_env.Dir2},
64 #else
65       {static_env.SymlinkToDir / "dir2/../dir2/DNE/..", static_env.Dir2 / ""},
66 #endif
67       {static_env.SymlinkToDir / "dir2/dir3/../DNE/DNE2", static_env.Dir2 / "DNE/DNE2"},
68       {static_env.Dir / "../dir1", static_env.Dir},
69       {static_env.Dir / "./.", static_env.Dir},
70       {static_env.Dir / "DNE/../foo", static_env.Dir / "foo"}
71   };
72   // clang-format on
73   for (auto& TC : TestCases) {
74     fs::path p      = TC.input;
75     fs::path expect = TC.expect;
76     expect.make_preferred();
77 
78     {
79       const fs::path output = fs::weakly_canonical(p);
80       TEST_REQUIRE(PathEq(output, expect),
81                    TEST_WRITE_CONCATENATED(
82                        "Input: ", TC.input.string(), "\nExpected: ", expect.string(), "\nOutput: ", output.string()));
83     }
84 
85     // Test the error_code variant
86     {
87       std::error_code ec;
88       const fs::path output_c = fs::weakly_canonical(p, ec);
89 
90       TEST_REQUIRE(PathEq(output_c, expect),
91                    TEST_WRITE_CONCATENATED(
92                        "Input: ", TC.input.string(), "\nExpected: ", expect.string(), "\nOutput: ", output_c.string()));
93     }
94   }
95   return 0;
96 }
97