xaizek / vifm (License: GPLv2+) (since 2018-12-07)
Vifm is a file manager with curses interface, which provides Vi[m]-like environment for managing objects within file systems, extended with some useful ideas from mutt.
<root> / tests / fileops / clone_files.c (38c6aae07a6d2e539f080812091e19d360ef2c0b) (6,797B) (mode 100644) [raw]
#include <stic.h>

#include <sys/stat.h> /* chmod() */
#include <unistd.h> /* chdir() unlink() */

#include <limits.h> /* INT_MAX */

#include <test-utils.h>

#include "../../src/cfg/config.h"
#include "../../src/utils/fs.h"
#include "../../src/filelist.h"
#include "../../src/fops_misc.h"

static char *saved_cwd;

SETUP()
{
	saved_cwd = save_cwd();
	assert_success(chdir(SANDBOX_PATH));

	view_setup(&lwin);
	assert_non_null(get_cwd(lwin.curr_dir, sizeof(lwin.curr_dir)));

	cfg.use_trash = 0;
}

TEARDOWN()
{
	view_teardown(&lwin);
	restore_cwd(saved_cwd);
}

TEST(files_are_cloned)
{
	create_file("a");
	populate_dir_list(&lwin, 0);
	lwin.dir_entry[0].marked = 1;

	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	(void)fops_clone(&lwin, NULL, 0, 0, 2);

	assert_success(unlink("a"));
	assert_success(unlink("a(1)"));
	assert_success(unlink("a(2)"));
	assert_success(unlink("a(3)"));
}

TEST(files_are_cloned_with_custom_name)
{
	char *names[] = { "b" };

	create_file("a");
	populate_dir_list(&lwin, 0);
	lwin.dir_entry[0].marked = 1;

	(void)fops_clone(&lwin, names, 1, 0, 1);

	assert_success(unlink("a"));
	assert_success(unlink("b"));
}

TEST(files_are_cloned_according_to_tree_structure)
{
	create_dir("dir");
	create_file("dir/a");

	/* Clone at the top level. */

	flist_load_tree(&lwin, ".", INT_MAX);
	lwin.list_pos = 0;

	lwin.dir_entry[0].marked = 1;
	lwin.dir_entry[1].marked = 0;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_success(unlink("dir(1)/a"));
	assert_success(rmdir("dir(1)"));

	/* Clone at nested level. */

	flist_load_tree(&lwin, ".", INT_MAX);
	lwin.list_pos = 0;

	lwin.dir_entry[0].marked = 0;
	lwin.dir_entry[1].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_success(unlink("dir/a(1)"));

	/* Clone at both levels. */

	flist_load_tree(&lwin, ".", INT_MAX);
	lwin.list_pos = 0;

	lwin.dir_entry[0].marked = 1;
	lwin.dir_entry[1].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_success(unlink("dir(1)/a"));
	assert_success(rmdir("dir(1)"));
	assert_success(unlink("dir/a(1)"));

	/* Cloning same file twice. */

	flist_load_tree(&lwin, ".", INT_MAX);
	lwin.list_pos = 0;
	lwin.dir_entry[1].marked = 1;
	assert_string_equal("a", lwin.dir_entry[1].name);
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	populate_dir_list(&lwin, 1);
	lwin.list_pos = 0;
	lwin.dir_entry[1].marked = 1;
	assert_string_equal("a", lwin.dir_entry[1].name);
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_success(unlink("dir/a(1)"));
	assert_success(unlink("dir/a(2)"));

	assert_success(unlink("dir/a"));
	assert_success(rmdir("dir"));
}

TEST(cloning_does_not_work_in_custom_view)
{
	char *names[] = { "a-clone" };

	create_file("do-not-clone-me");

	/* Without specifying new name. */

	flist_custom_start(&lwin, "test");
	flist_custom_add(&lwin, "do-not-clone-me");
	assert_true(flist_custom_finish(&lwin, CV_REGULAR, 0) == 0);

	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_failure(unlink("do-not-clone-me(1)"));

	/* Must recreate custom view to account for cloning renaming a file entry. */

	/* With name specified. */

	flist_custom_start(&lwin, "test");
	flist_custom_add(&lwin, "do-not-clone-me");
	assert_true(flist_custom_finish(&lwin, CV_REGULAR, 0) == 0);

	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, names, 1, 0, 1);
	assert_failure(unlink("a-clone"));

	assert_success(unlink("do-not-clone-me"));
}

TEST(cloning_does_not_work_in_custom_tree)
{
	char *names[] = { "a-clone" };

	create_file("do-not-clone-me");

	/* Without specifying new name. */

	flist_custom_start(&lwin, "test");
	flist_custom_add(&lwin, "do-not-clone-me");
	assert_true(flist_custom_finish(&lwin, CV_REGULAR, 0) == 0);

	assert_success(flist_load_tree(&lwin, ".", INT_MAX));
	assert_int_equal(CV_CUSTOM_TREE, lwin.custom.type);

	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_failure(unlink("do-not-clone-me(1)"));

	/* Must recreate custom view to account for cloning renaming a file entry. */

	/* With name specified. */

	flist_custom_start(&lwin, "test");
	flist_custom_add(&lwin, "do-not-clone-me");
	assert_true(flist_custom_finish(&lwin, CV_REGULAR, 0) == 0);

	assert_success(flist_load_tree(&lwin, ".", INT_MAX));
	assert_int_equal(CV_CUSTOM_TREE, lwin.custom.type);

	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, names, 1, 0, 1);
	assert_failure(unlink("a-clone"));

	assert_success(unlink("do-not-clone-me"));
}

TEST(cloning_folded_dir)
{
	create_dir("dir");
	create_file("dir/a");

	flist_load_tree(&lwin, ".", INT_MAX);
	assert_int_equal(2, lwin.list_rows);
	lwin.list_pos = 0;

	flist_toggle_fold(&lwin);
	populate_dir_list(&lwin, /*reload=*/1);
	assert_int_equal(1, lwin.list_rows);

	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);

	populate_dir_list(&lwin, /*reload=*/1);
	assert_int_equal(2, lwin.list_rows);

	lwin.list_pos = 1;
	flist_toggle_fold(&lwin);
	lwin.list_pos = 0;
	flist_toggle_fold(&lwin);
	populate_dir_list(&lwin, /*reload=*/1);
	assert_int_equal(4, lwin.list_rows);

	assert_success(unlink("dir(1)/a"));
	assert_success(unlink("dir/a"));
	assert_success(rmdir("dir"));
	assert_success(rmdir("dir(1)"));
}

TEST(cloning_dir_clones_child_folds)
{
	create_dir("dir");
	create_dir("dir/sub");
	create_file("dir/sub/a");

	flist_load_tree(&lwin, ".", INT_MAX);
	assert_int_equal(3, lwin.list_rows);

	lwin.list_pos = 1;
	flist_toggle_fold(&lwin);
	populate_dir_list(&lwin, /*reload=*/1);
	assert_int_equal(2, lwin.list_rows);

	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);

	populate_dir_list(&lwin, /*reload=*/1);
	assert_int_equal(4, lwin.list_rows);

	lwin.list_pos = 3;
	flist_toggle_fold(&lwin);
	populate_dir_list(&lwin, /*reload=*/1);
	assert_int_equal(5, lwin.list_rows);

	remove_file("dir(1)/sub/a");
	remove_dir("dir(1)/sub");
	remove_dir("dir(1)");

	remove_file("dir/sub/a");
	remove_dir("dir/sub");
	remove_dir("dir");
}

TEST(cloning_of_broken_symlink, IF(not_windows))
{
	assert_success(make_symlink("no-such-file", "broken-link"));

	flist_load_tree(&lwin, ".", INT_MAX);

	/* Without specifying new name. */
	lwin.dir_entry[0].marked = 1;
	(void)fops_clone(&lwin, NULL, 0, 0, 1);
	assert_success(unlink("broken-link(1)"));

	assert_success(unlink("broken-link"));
}

TEST(cloning_is_aborted_if_we_can_not_read_a_file, IF(regular_unix_user))
{
	create_file("can-read");
	create_file("can-not-read");
	assert_success(chmod("can-not-read", 0000));
	populate_dir_list(&lwin, 0);

	lwin.dir_entry[0].marked = 1;
	lwin.dir_entry[1].marked = 1;
	assert_int_equal(0, fops_clone(&lwin, NULL, 0, 0, 1));

	assert_success(unlink("can-read"));
	assert_success(unlink("can-not-read"));
}

/* vim: set tabstop=2 softtabstop=2 shiftwidth=2 noexpandtab cinoptions-=(0 : */
/* vim: set cinoptions+=t0 : */
Hints

Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://code.reversed.top/user/xaizek/vifm

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@code.reversed.top/user/xaizek/vifm

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a pull request:
... clone the repository ...
... make some changes and some commits ...
git push origin master