//===----------------------------------------------------------------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is dual licensed under the MIT and the University of Illinois Open
// Source Licenses. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
// UNSUPPORTED: c++98, c++03
// <experimental/filesystem>
// file_status status(const path& p);
// file_status status(const path& p, error_code& ec) noexcept;
#include <experimental/filesystem>
#include "test_macros.h"
#include "rapid-cxx-test.hpp"
#include "filesystem_test_helper.hpp"
using namespace std::experimental::filesystem;
TEST_SUITE(filesystem_status_test_suite)
TEST_CASE(signature_test)
{
const path p; ((void)p);
std::error_code ec; ((void)ec);
ASSERT_NOT_NOEXCEPT(status(p));
ASSERT_NOEXCEPT(status(p, ec));
}
TEST_CASE(test_status_not_found)
{
const std::error_code expect_ec =
std::make_error_code(std::errc::no_such_file_or_directory);
const path cases[] {
StaticEnv::DNE,
StaticEnv::BadSymlink
};
for (auto& p : cases) {
std::error_code ec = std::make_error_code(std::errc::address_in_use);
// test non-throwing overload.
file_status st = status(p, ec);
TEST_CHECK(ec == expect_ec);
TEST_CHECK(st.type() == file_type::not_found);
TEST_CHECK(st.permissions() == perms::unknown);
// test throwing overload. It should not throw even though it reports
// that the file was not found.
TEST_CHECK_NO_THROW(st = status(p));
TEST_CHECK(st.type() == file_type::not_found);
TEST_CHECK(st.permissions() == perms::unknown);
}
}
TEST_CASE(test_status_cannot_resolve)
{
scoped_test_env env;
const path dir = env.create_dir("dir");
const path file = env.create_file("dir/file", 42);
const path sym = env.create_symlink("dir/file", "sym");
permissions(dir, perms::none);
const std::error_code set_ec =
std::make_error_code(std::errc::address_in_use);
const std::error_code perm_ec =
std::make_error_code(std::errc::permission_denied);
const std::error_code name_too_long_ec =
std::make_error_code(std::errc::filename_too_long);
struct TestCase {
path p;
std::error_code expect_ec;
} const TestCases[] = {
{file, perm_ec},
{sym, perm_ec},
{path(std::string(2500, 'a')), name_too_long_ec}
};
for (auto& TC : TestCases)
{
{ // test non-throwing case
std::error_code ec = set_ec;
file_status st = status(TC.p, ec);
TEST_CHECK(ec == TC.expect_ec);
TEST_CHECK(st.type() == file_type::none);
TEST_CHECK(st.permissions() == perms::unknown);
}
#ifndef TEST_HAS_NO_EXCEPTIONS
{ // test throwing case
try {
status(TC.p);
} catch (filesystem_error const& err) {
TEST_CHECK(err.path1() == TC.p);
TEST_CHECK(err.path2() == "");
TEST_CHECK(err.code() == TC.expect_ec);
}
}
#endif
}
}
TEST_CASE(status_file_types_test)
{
scoped_test_env env;
struct TestCase {
path p;
file_type expect_type;
} cases[] = {
{StaticEnv::File, file_type::regular},
{StaticEnv::SymlinkToFile, file_type::regular},
{StaticEnv::Dir, file_type::directory},
{StaticEnv::SymlinkToDir, file_type::directory},
// Block files tested elsewhere
{StaticEnv::CharFile, file_type::character},
#if !defined(__APPLE__) && !defined(__FreeBSD__) // No support for domain sockets
{env.create_socket("socket"), file_type::socket},
#endif
{env.create_fifo("fifo"), file_type::fifo}
};
for (const auto& TC : cases) {
// test non-throwing case
std::error_code ec = std::make_error_code(std::errc::address_in_use);
file_status st = status(TC.p, ec);
TEST_CHECK(!ec);
TEST_CHECK(st.type() == TC.expect_type);
TEST_CHECK(st.permissions() != perms::unknown);
// test throwing case
TEST_REQUIRE_NO_THROW(st = status(TC.p));
TEST_CHECK(st.type() == TC.expect_type);
TEST_CHECK(st.permissions() != perms::unknown);
}
}
TEST_CASE(test_block_file)
{
const path possible_paths[] = {
"/dev/drive0", // Apple
"/dev/sda",
"/dev/loop0"
};
path p;
for (const path& possible_p : possible_paths) {
std::error_code ec;
if (exists(possible_p, ec)) {
p = possible_p;
break;
}
}
if (p == path{}) {
TEST_UNSUPPORTED();
}
// test non-throwing case
std::error_code ec = std::make_error_code(std::errc::address_in_use);
file_status st = status(p, ec);
TEST_CHECK(!ec);
TEST_CHECK(st.type() == file_type::block);
TEST_CHECK(st.permissions() != perms::unknown);
// test throwing case
TEST_REQUIRE_NO_THROW(st = status(p));
TEST_CHECK(st.type() == file_type::block);
TEST_CHECK(st.permissions() != perms::unknown);
}
TEST_SUITE_END()