316 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			316 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
//                     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
 | 
						|
 | 
						|
// <filesystem>
 | 
						|
 | 
						|
// void copy(const path& from, const path& to);
 | 
						|
// void copy(const path& from, const path& to, error_code& ec);
 | 
						|
// void copy(const path& from, const path& to, copy_options options);
 | 
						|
// void copy(const path& from, const path& to, copy_options options,
 | 
						|
//           error_code& ec);
 | 
						|
 | 
						|
#include "filesystem_include.hpp"
 | 
						|
#include <type_traits>
 | 
						|
#include <cstddef>
 | 
						|
#include <cassert>
 | 
						|
 | 
						|
#include "test_macros.h"
 | 
						|
#include "rapid-cxx-test.hpp"
 | 
						|
#include "filesystem_test_helper.hpp"
 | 
						|
 | 
						|
using namespace fs;
 | 
						|
 | 
						|
using CO = fs::copy_options;
 | 
						|
 | 
						|
TEST_SUITE(filesystem_copy_test_suite)
 | 
						|
 | 
						|
TEST_CASE(signature_test)
 | 
						|
{
 | 
						|
    const path p; ((void)p);
 | 
						|
    std::error_code ec; ((void)ec);
 | 
						|
    const copy_options opts{}; ((void)opts);
 | 
						|
    ASSERT_NOT_NOEXCEPT(fs::copy(p, p));
 | 
						|
    ASSERT_NOT_NOEXCEPT(fs::copy(p, p, ec));
 | 
						|
    ASSERT_NOT_NOEXCEPT(copy(p, p, opts));
 | 
						|
    ASSERT_NOT_NOEXCEPT(copy(p, p, opts, ec));
 | 
						|
}
 | 
						|
 | 
						|
// There are 4 cases is the proposal for absolute path.
 | 
						|
// Each scope tests one of the cases.
 | 
						|
TEST_CASE(test_error_reporting)
 | 
						|
{
 | 
						|
    auto checkThrow = [](path const& f, path const& t, const std::error_code& ec)
 | 
						|
    {
 | 
						|
#ifndef TEST_HAS_NO_EXCEPTIONS
 | 
						|
        try {
 | 
						|
            fs::copy(f, t);
 | 
						|
            return false;
 | 
						|
        } catch (filesystem_error const& err) {
 | 
						|
            return err.path1() == f
 | 
						|
                && err.path2() == t
 | 
						|
                && err.code() == ec;
 | 
						|
        }
 | 
						|
#else
 | 
						|
        ((void)f); ((void)t); ((void)ec);
 | 
						|
        return true;
 | 
						|
#endif
 | 
						|
    };
 | 
						|
 | 
						|
    scoped_test_env env;
 | 
						|
    const path file = env.create_file("file1", 42);
 | 
						|
    const path dir = env.create_dir("dir");
 | 
						|
    const path fifo = env.create_fifo("fifo");
 | 
						|
    TEST_REQUIRE(is_other(fifo));
 | 
						|
 | 
						|
    const auto test_ec = GetTestEC();
 | 
						|
 | 
						|
    // !exists(f)
 | 
						|
    {
 | 
						|
        std::error_code ec = test_ec;
 | 
						|
        const path f = StaticEnv::DNE;
 | 
						|
        const path t = env.test_root;
 | 
						|
        fs::copy(f, t, ec);
 | 
						|
        TEST_REQUIRE(ec);
 | 
						|
        TEST_REQUIRE(ec != test_ec);
 | 
						|
        TEST_CHECK(checkThrow(f, t, ec));
 | 
						|
    }
 | 
						|
    { // equivalent(f, t) == true
 | 
						|
        std::error_code ec = test_ec;
 | 
						|
        fs::copy(file, file, ec);
 | 
						|
        TEST_REQUIRE(ec);
 | 
						|
        TEST_REQUIRE(ec != test_ec);
 | 
						|
        TEST_CHECK(checkThrow(file, file, ec));
 | 
						|
    }
 | 
						|
    { // is_directory(from) && is_file(to)
 | 
						|
        std::error_code ec = test_ec;
 | 
						|
        fs::copy(dir, file, ec);
 | 
						|
        TEST_REQUIRE(ec);
 | 
						|
        TEST_REQUIRE(ec != test_ec);
 | 
						|
        TEST_CHECK(checkThrow(dir, file, ec));
 | 
						|
    }
 | 
						|
    { // is_other(from)
 | 
						|
        std::error_code ec = test_ec;
 | 
						|
        fs::copy(fifo, dir, ec);
 | 
						|
        TEST_REQUIRE(ec);
 | 
						|
        TEST_REQUIRE(ec != test_ec);
 | 
						|
        TEST_CHECK(checkThrow(fifo, dir, ec));
 | 
						|
    }
 | 
						|
    { // is_other(to)
 | 
						|
        std::error_code ec = test_ec;
 | 
						|
        fs::copy(file, fifo, ec);
 | 
						|
        TEST_REQUIRE(ec);
 | 
						|
        TEST_REQUIRE(ec != test_ec);
 | 
						|
        TEST_CHECK(checkThrow(file, fifo, ec));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE(from_is_symlink)
 | 
						|
{
 | 
						|
    scoped_test_env env;
 | 
						|
    const path file = env.create_file("file", 42);
 | 
						|
    const path symlink = env.create_symlink(file, "sym");
 | 
						|
    const path dne = env.make_env_path("dne");
 | 
						|
 | 
						|
    { // skip symlinks
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(symlink, dne, copy_options::skip_symlinks, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(!exists(dne));
 | 
						|
    }
 | 
						|
    {
 | 
						|
        const path dest = env.make_env_path("dest");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(symlink, dest, copy_options::copy_symlinks, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(exists(dest));
 | 
						|
        TEST_CHECK(is_symlink(dest));
 | 
						|
    }
 | 
						|
    { // copy symlink but target exists
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(symlink, file, copy_options::copy_symlinks, ec);
 | 
						|
        TEST_CHECK(ec);
 | 
						|
        TEST_CHECK(ec != GetTestEC());
 | 
						|
    }
 | 
						|
    { // create symlinks but target exists
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(symlink, file, copy_options::create_symlinks, ec);
 | 
						|
        TEST_CHECK(ec);
 | 
						|
        TEST_CHECK(ec != GetTestEC());
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE(from_is_regular_file)
 | 
						|
{
 | 
						|
    scoped_test_env env;
 | 
						|
    const path file = env.create_file("file", 42);
 | 
						|
    const path dir = env.create_dir("dir");
 | 
						|
    { // skip copy because of directory
 | 
						|
        const path dest = env.make_env_path("dest1");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(file, dest, CO::directories_only, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(!exists(dest));
 | 
						|
    }
 | 
						|
    { // create symlink to file
 | 
						|
        const path dest = env.make_env_path("sym");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(file, dest, CO::create_symlinks, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(is_symlink(dest));
 | 
						|
        TEST_CHECK(equivalent(file, canonical(dest)));
 | 
						|
    }
 | 
						|
    { // create hard link to file
 | 
						|
        const path dest = env.make_env_path("hardlink");
 | 
						|
        TEST_CHECK(hard_link_count(file) == 1);
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(file, dest, CO::create_hard_links, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(exists(dest));
 | 
						|
        TEST_CHECK(hard_link_count(file) == 2);
 | 
						|
    }
 | 
						|
    { // is_directory(t)
 | 
						|
        const path dest_dir = env.create_dir("dest_dir");
 | 
						|
        const path expect_dest = dest_dir / file.filename();
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(file, dest_dir, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(is_regular_file(expect_dest));
 | 
						|
    }
 | 
						|
    { // otherwise copy_file(from, to, ...)
 | 
						|
        const path dest = env.make_env_path("file_copy");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(file, dest, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(is_regular_file(dest));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE(from_is_directory)
 | 
						|
{
 | 
						|
    struct FileInfo {
 | 
						|
        path filename;
 | 
						|
        std::size_t size;
 | 
						|
    };
 | 
						|
    const FileInfo files[] = {
 | 
						|
        {"file1", 0},
 | 
						|
        {"file2", 42},
 | 
						|
        {"file3", 300}
 | 
						|
    };
 | 
						|
    scoped_test_env env;
 | 
						|
    const path dir = env.create_dir("dir");
 | 
						|
    const path nested_dir_name = "dir2";
 | 
						|
    const path nested_dir = env.create_dir("dir/dir2");
 | 
						|
 | 
						|
    for (auto& FI : files) {
 | 
						|
        env.create_file(dir / FI.filename, FI.size);
 | 
						|
        env.create_file(nested_dir / FI.filename, FI.size);
 | 
						|
    }
 | 
						|
    { // test for non-existent directory
 | 
						|
        const path dest = env.make_env_path("dest_dir1");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(dir, dest, ec);
 | 
						|
        TEST_REQUIRE(!ec);
 | 
						|
        TEST_CHECK(is_directory(dest));
 | 
						|
        for (auto& FI : files) {
 | 
						|
            path created = dest / FI.filename;
 | 
						|
            TEST_CHECK(is_regular_file(created));
 | 
						|
            TEST_CHECK(file_size(created) == FI.size);
 | 
						|
        }
 | 
						|
        TEST_CHECK(!is_directory(dest / nested_dir_name));
 | 
						|
    }
 | 
						|
    { // test for existing directory
 | 
						|
        const path dest = env.create_dir("dest_dir2");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(dir, dest, ec);
 | 
						|
        TEST_REQUIRE(!ec);
 | 
						|
        TEST_CHECK(is_directory(dest));
 | 
						|
        for (auto& FI : files) {
 | 
						|
            path created = dest / FI.filename;
 | 
						|
            TEST_CHECK(is_regular_file(created));
 | 
						|
            TEST_CHECK(file_size(created) == FI.size);
 | 
						|
        }
 | 
						|
        TEST_CHECK(!is_directory(dest / nested_dir_name));
 | 
						|
    }
 | 
						|
    { // test recursive copy
 | 
						|
        const path dest = env.make_env_path("dest_dir3");
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(dir, dest, CO::recursive, ec);
 | 
						|
        TEST_REQUIRE(!ec);
 | 
						|
        TEST_CHECK(is_directory(dest));
 | 
						|
        const path nested_dest = dest / nested_dir_name;
 | 
						|
        TEST_REQUIRE(is_directory(nested_dest));
 | 
						|
        for (auto& FI : files) {
 | 
						|
            path created = dest / FI.filename;
 | 
						|
            path nested_created = nested_dest / FI.filename;
 | 
						|
            TEST_CHECK(is_regular_file(created));
 | 
						|
            TEST_CHECK(file_size(created) == FI.size);
 | 
						|
            TEST_CHECK(is_regular_file(nested_created));
 | 
						|
            TEST_CHECK(file_size(nested_created) == FI.size);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE(test_copy_symlinks_to_symlink_dir)
 | 
						|
{
 | 
						|
    scoped_test_env env;
 | 
						|
    const path file1 = env.create_file("file1", 42);
 | 
						|
    const path file2 = env.create_file("file2", 101);
 | 
						|
    const path file2_sym = env.create_symlink(file2, "file2_sym");
 | 
						|
    const path dir = env.create_dir("dir");
 | 
						|
    const path dir_sym = env.create_symlink(dir, "dir_sym");
 | 
						|
    {
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(file1, dir_sym, copy_options::copy_symlinks, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        const path dest = env.make_env_path("dir/file1");
 | 
						|
        TEST_CHECK(exists(dest));
 | 
						|
        TEST_CHECK(!is_symlink(dest));
 | 
						|
        TEST_CHECK(file_size(dest) == 42);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
TEST_CASE(test_dir_create_symlink)
 | 
						|
{
 | 
						|
    scoped_test_env env;
 | 
						|
    const path dir = env.create_dir("dir1");
 | 
						|
    const path dest = env.make_env_path("dne");
 | 
						|
    {
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(dir, dest, copy_options::create_symlinks, ec);
 | 
						|
        TEST_CHECK(ec == std::make_error_code(std::errc::is_a_directory));
 | 
						|
        TEST_CHECK(!exists(dest));
 | 
						|
        TEST_CHECK(!is_symlink(dest));
 | 
						|
    }
 | 
						|
    {
 | 
						|
        std::error_code ec = GetTestEC();
 | 
						|
        fs::copy(dir, dest, copy_options::create_symlinks|copy_options::recursive, ec);
 | 
						|
        TEST_CHECK(ec == std::make_error_code(std::errc::is_a_directory));
 | 
						|
        TEST_CHECK(!exists(dest));
 | 
						|
        TEST_CHECK(!is_symlink(dest));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE(test_otherwise_no_effects_clause)
 | 
						|
{
 | 
						|
    scoped_test_env env;
 | 
						|
    const path dir = env.create_dir("dir1");
 | 
						|
    { // skip copy because of directory
 | 
						|
        const path dest = env.make_env_path("dest1");
 | 
						|
        std::error_code ec;
 | 
						|
        fs::copy(dir, dest, CO::directories_only, ec);
 | 
						|
        TEST_CHECK(!ec);
 | 
						|
        TEST_CHECK(!exists(dest));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
TEST_SUITE_END()
 |