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 // bool is_symlink(file_status s) noexcept
17 // bool is_symlink(path const& p);
18 // bool is_symlink(path const& p, std::error_code& ec) noexcept;
19 
20 #include <filesystem>
21 #include <type_traits>
22 #include <cassert>
23 
24 #include "assert_macros.h"
25 #include "test_macros.h"
26 #include "filesystem_test_helper.h"
27 namespace fs = std::filesystem;
28 using namespace fs;
29 
signature_test()30 static void signature_test()
31 {
32     file_status s; ((void)s);
33     const path p; ((void)p);
34     std::error_code ec; ((void)ec);
35     ASSERT_NOEXCEPT(is_symlink(s));
36     ASSERT_NOEXCEPT(is_symlink(p, ec));
37     ASSERT_NOT_NOEXCEPT(is_symlink(p));
38 }
39 
is_symlink_status_test()40 static void is_symlink_status_test()
41 {
42     struct TestCase {
43         file_type type;
44         bool expect;
45     };
46     const TestCase testCases[] = {
47         {file_type::none, false},
48         {file_type::not_found, false},
49         {file_type::regular, false},
50         {file_type::directory, false},
51         {file_type::symlink, true},
52         {file_type::block, false},
53         {file_type::character, false},
54         {file_type::fifo, false},
55         {file_type::socket, false},
56         {file_type::unknown, false}
57     };
58     for (auto& TC : testCases) {
59         file_status s(TC.type);
60         assert(is_symlink(s) == TC.expect);
61     }
62 }
63 
static_env_test()64 static void static_env_test()
65 {
66     static_test_env static_env;
67     struct TestCase {
68         path p;
69         bool expect;
70     };
71     const TestCase testCases[] = {
72         {static_env.File, false},
73         {static_env.Dir, false},
74         {static_env.SymlinkToFile, true},
75         {static_env.SymlinkToDir, true},
76         {static_env.BadSymlink, true}
77     };
78     for (auto& TC : testCases) {
79         assert(is_symlink(TC.p) == TC.expect);
80     }
81 }
82 
test_exist_not_found()83 static void test_exist_not_found()
84 {
85     static_test_env static_env;
86     const path p = static_env.DNE;
87     assert(is_symlink(p) == false);
88     std::error_code ec;
89     assert(is_symlink(p, ec) == false);
90     assert(ec);
91 }
92 
test_is_symlink_fails()93 static void test_is_symlink_fails()
94 {
95     scoped_test_env env;
96 #ifdef _WIN32
97     // Windows doesn't support setting perms::none to trigger failures
98     // reading directories; test using a special inaccessible directory
99     // instead.
100     const path p = GetWindowsInaccessibleDir();
101     if (p.empty())
102         return;
103 #else
104     const path dir = env.create_dir("dir");
105     const path p = env.create_file("dir/file", 42);
106     permissions(dir, perms::none);
107 #endif
108 
109     std::error_code ec;
110     assert(is_symlink(p, ec) == false);
111     assert(ec);
112 
113     TEST_THROWS_TYPE(filesystem_error, is_symlink(p));
114 }
115 
main(int,char **)116 int main(int, char**) {
117     signature_test();
118     is_symlink_status_test();
119     static_env_test();
120     test_exist_not_found();
121     test_is_symlink_fails();
122 
123     return 0;
124 }
125