Add more recurse simple program unit tests.
Add the tree simple program unit tests.
Implement the tree delete operation.
Relocate the parent directory pre-process to a loop that is processed after all other files and directories are removed.
build_sources_program test-remove.c
build_sources_program test-remove-print_help.c test-remove-print_version.c
build_sources_program test-remove-directory_no_args.c test-remove-regular_no_args.c
-build_sources_program test-remove-directory_recurse_simple.c
+build_sources_program test-remove-directory_recurse_simple.c test-remove-directory_tree_simple.c
build_script no
build_shared yes
}
#endif // _di_kt_remove_operate_file_directory_action_
+#ifndef _di_kt_remove_operate_file_parent_
+ void kt_remove_operate_file_parent(kt_remove_main_t * const main, const f_string_static_t path) {
+
+ if (!main) return;
+
+ if (!path.used) {
+ main->setting.state.status = F_data_not;
+
+ return;
+ }
+
+ if (kt_remove_signal_check(main)) return;
+
+ const uint16_t flag_operate = kt_remove_preprocess_file(main, path, kt_remove_flag_file_operate_parent_d);
+
+ if (!kt_remove_operate_shall_remove(flag_operate) || (main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+
+ if (F_status_is_error_not(main->setting.state.status) && !(flag_operate & kt_remove_flag_file_operate_processed_d)) {
+ main->setting.state.status = kt_remove_operate_file_remove(main, path, flag_operate);
+ }
+
+ if (F_status_is_error_not(main->setting.state.status)) {
+ kt_remove_operate_memory_save(main, path, flag_operate);
+
+ if (F_status_is_error(main->setting.state.status)) {
+ kt_remove_print_error_file(&main->program.error, macro_kt_remove_f(kt_remove_operate_memory_save), path, f_file_operation_process_s, fll_error_file_type_path_e);
+ }
+ }
+
+ if (F_status_is_error_not(main->setting.state.status)) {
+ main->setting.state.status = F_okay;
+ }
+ }
+#endif // _di_kt_remove_operate_file_parent_
+
#ifndef _di_kt_remove_operate_file_recurse_handle_
void kt_remove_operate_file_recurse_handle(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint32_t flag) {
#endif // _di_kt_remove_operate_file_directory_action_
/**
+ * Operate on a single parent directory (from the tree array).
+ *
+ * @param main
+ * The main program and settings data.
+ *
+ * Must not be NULL.
+ *
+ * This alters main.setting.state.status:
+ * F_yes on success and file remove.
+ * F_no on success and file not removed.
+ * F_data_not on success but path is an empty string.
+ *
+ * F_no (with error bit set) on file not removed due to failure.
+ *
+ * Errors (with error bit) from: f_string_dynamic_append().
+ *
+ * Errors (with error bit) from: kt_remove_operate_file_directory().
+ * Errors (with error bit) from: kt_remove_operate_file_remove_delete().
+ * Errors (with error bit) from: kt_remove_operate_file_remove().
+ * Errors (with error bit) from: kt_remove_preprocess_file().
+ * @param path
+ * The path to the file to operate on.
+ *
+ * This should always be TRUE when calling from the top level.
+ * This should always be FALSE if calling from within a fl_directory_do() callback.
+ * This is because fl_directory_do() handles directory traversal and processing.
+ *
+ * @see f_string_dynamic_append()
+ *
+ * @see kt_remove_operate_file_directory()
+ * @see kt_remove_operate_file_remove_delete()
+ * @see kt_remove_operate_file_remove()
+ * @see kt_remove_preprocess_file()
+ */
+#ifndef _di_kt_remove_operate_file_parent_
+ extern void kt_remove_operate_file_parent(kt_remove_main_t * const main, const f_string_static_t path);
+#endif // _di_kt_remove_operate_file_parent_
+
+/**
* Handle errors while performing directory recurse for a single file operation action.
*
* @param recurse
}
++main->cache.tree.used;
-
- f_print_dynamic(f_string_eol_s, main->program.output.to);
-
- // Pre-process parent directories and set parent flag to prevent potential recursion.
- kt_remove_preprocess_file(main, main->cache.tree.array[main->cache.tree.used - 1], kt_remove_flag_file_operate_parent_d);
}
}
else if (F_status_is_error(main->setting.state.status)) {
main->setting.state.status = F_okay;
- for (f_number_unsigned_t i = 0; i < main->setting.files.used; ++i) {
+ f_number_unsigned_t i = 0;
+
+ for (; i < main->setting.files.used; ++i) {
kt_remove_operate_file(main, main->setting.files.array[i]);
f_print_dynamic(f_string_eol_s, main->program.output.to);
}
- if (F_status_is_error(main->setting.state.status)) break;
+ if (F_status_is_error(main->setting.state.status)) return;
} // for
+
+ if (main->cache.tree.used) {
+ for (i = 0; i < main->cache.tree.used; ++i) {
+
+ if (main->setting.flag & kt_remove_main_flag_simulate_d) {
+ f_print_dynamic(f_string_eol_s, main->program.output.to);
+ }
+
+ kt_remove_operate_file_parent(main, main->cache.tree.array[i]);
+ if (F_status_is_error(main->setting.state.status)) return;
+ } // for
+ }
}
#endif // _di_kt_remove_process_normal_operate_
for (uint32_t i = 0; i < total; ++i) {
f_string_static_t * const child_path = mock_ptr_type(f_string_static_t *);
+ f_string_static_t * const child_name = mock_ptr_type(f_string_static_t *);
recurse->path = child_path ? *child_path : f_string_empty_s;
- recurse->action(recurse, recurse->path, mock_type(uint32_t));
+ recurse->action(recurse, child_name ? *child_name : f_string_empty_s, mock_type(uint32_t));
} // for
}
else {
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_false); // kt_remove_flag_file_operate_empty_d is set.
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_false); // kt_remove_flag_file_operate_empty_d is set.
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_false); // kt_remove_flag_file_operate_empty_d is set.
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_false); // kt_remove_flag_file_operate_empty_d is set.
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_false); // kt_remove_flag_file_operate_empty_d is set.
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_false); // kt_remove_flag_file_operate_empty_d is set.
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
will_return(__wrap_f_directory_empty, F_true);
will_return(__wrap_fl_directory_do, 1);
will_return(__wrap_fl_directory_do, &file);
+ will_return(__wrap_fl_directory_do, &file);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
// This will fail if f_directory_remove() is not called, therefore success here means f_directory_remove has been called.
mock_unwrap = 0;
- const f_string_static_t parent = macro_f_string_static_t_initialize_1("to_remove", 0, 9);
+ const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9);
const f_string_static_t child_1 = macro_f_string_static_t_initialize_1("child_1", 0, 7);
- struct stat stat_parent;
+ struct stat stat_directory;
struct stat stat_regular;
{
- const f_string_t argv[] = { "mocked_main", parent.string, "-r", 0 };
+ const f_string_t argv[] = { "mocked_main", target.string, "-r", 0 };
- memset(&stat_parent, 0, sizeof(struct stat));
+ memset(&stat_directory, 0, sizeof(struct stat));
memset(&stat_regular, 0, sizeof(struct stat));
- stat_parent.st_mode = F_file_mode_all_d | F_file_type_directory_d;
+ stat_directory.st_mode = F_file_mode_all_d | F_file_type_directory_d;
stat_regular.st_mode = F_file_mode_all_d | F_file_type_regular_d;
// Pre-process file.
will_return(__wrap_f_file_exists, F_true);
will_return(__wrap_f_file_is, F_true); // A link, kt_remove_flag_file_operate_link_d is set.
- will_return(__wrap_f_file_stat, &stat_parent);
+ will_return(__wrap_f_file_stat, &stat_directory);
will_return(__wrap_f_file_stat, F_okay);
- // Parent Directory pre-processing, begin, perform directory listing action.
+ // Target Directory pre-processing, begin, perform directory listing action.
will_return(__wrap_f_directory_empty, F_false);
will_return(__wrap_fl_directory_do, 1);
// The first child pre-processing.
will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
+ will_return(__wrap_f_file_stat, &stat_regular);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Target Directory processing, begin.
+ will_return(__wrap_fl_directory_do, 2);
+
+ // The first child processing.
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_remove, F_okay);
+
+ // Target Directory processing, continue.
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
+ will_return(__wrap_f_directory_remove, F_okay);
+
+ const int result = kt_main_test__remove(3, argv, 0);
+
+ assert_int_equal(result, 0);
+ }
+}
+
+void test__kt_remove__directory_recurse_simple__one_child_one_exists_link_not(void **state) {
+
+ mock_unwrap = 0;
+
+ const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9);
+ const f_string_static_t child_1 = macro_f_string_static_t_initialize_1("child_1", 0, 7);
+
+ struct stat stat_directory;
+ struct stat stat_regular;
+
+ {
+ const f_string_t argv[] = { "mocked_main", target.string, "-r", 0 };
+
+ memset(&stat_directory, 0, sizeof(struct stat));
+ memset(&stat_regular, 0, sizeof(struct stat));
+
+ stat_directory.st_mode = F_file_mode_all_d | F_file_type_directory_d;
+ stat_regular.st_mode = F_file_mode_all_d | F_file_type_regular_d;
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
+ will_return(__wrap_f_file_stat, &stat_directory);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Target Directory pre-processing, begin, perform directory listing action.
+ will_return(__wrap_f_directory_empty, F_false);
+ will_return(__wrap_fl_directory_do, 1);
+
+ // The first child pre-processing.
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
will_return(__wrap_f_file_exists, F_true);
will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
will_return(__wrap_f_file_stat, &stat_regular);
will_return(__wrap_f_file_stat, F_okay);
- // Parent Directory processing, begin.
+ // Target Directory processing, begin.
will_return(__wrap_fl_directory_do, 2);
// The first child processing.
will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_remove, F_okay);
+
+ // Target Directory processing, continue.
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
+ will_return(__wrap_f_directory_remove, F_okay);
+
+ const int result = kt_main_test__remove(3, argv, 0);
+
+ assert_int_equal(result, 0);
+ }
+}
+
+void test__kt_remove__directory_recurse_simple__one_child_two_exists_link(void **state) {
+
+ mock_unwrap = 0;
+
+ const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9);
+ const f_string_static_t child_1 = macro_f_string_static_t_initialize_1("child_1", 0, 7);
+ const f_string_static_t child_2 = macro_f_string_static_t_initialize_1("child_2", 0, 7);
+
+ struct stat stat_directory;
+ struct stat stat_regular;
+
+ {
+ const f_string_t argv[] = { "mocked_main", target.string, "-r", 0 };
+
+ memset(&stat_directory, 0, sizeof(struct stat));
+ memset(&stat_regular, 0, sizeof(struct stat));
+
+ stat_directory.st_mode = F_file_mode_all_d | F_file_type_directory_d;
+ stat_regular.st_mode = F_file_mode_all_d | F_file_type_regular_d;
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_true); // A link, kt_remove_flag_file_operate_link_d is set.
+ will_return(__wrap_f_file_stat, &stat_directory);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Target Directory pre-processing, begin, perform directory listing action.
+ will_return(__wrap_f_directory_empty, F_false);
+ will_return(__wrap_fl_directory_do, 2);
+
+ // The first child pre-processing.
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
+ will_return(__wrap_f_file_stat, &stat_regular);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // The second child pre-processing.
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_true); // A link, kt_remove_flag_file_operate_link_d is set.
+ will_return(__wrap_f_file_stat, &stat_regular);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Target Directory processing, begin.
+ will_return(__wrap_fl_directory_do, 3);
+
+ // The first child processing.
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_remove, F_okay);
+
+ // The second child processing.
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_remove, F_okay);
+
+ // Target Directory processing, continue.
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
+ will_return(__wrap_f_directory_remove, F_okay);
+
+ const int result = kt_main_test__remove(3, argv, 0);
+
+ assert_int_equal(result, 0);
+ }
+}
+
+void test__kt_remove__directory_recurse_simple__one_child_two_exists_link_not(void **state) {
+
+ mock_unwrap = 0;
+
+ const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9);
+ const f_string_static_t child_1 = macro_f_string_static_t_initialize_1("child_1", 0, 7);
+ const f_string_static_t child_2 = macro_f_string_static_t_initialize_1("child_2", 0, 7);
+
+ struct stat stat_directory;
+ struct stat stat_regular;
+
+ {
+ const f_string_t argv[] = { "mocked_main", target.string, "-r", 0 };
+
+ memset(&stat_directory, 0, sizeof(struct stat));
+ memset(&stat_regular, 0, sizeof(struct stat));
+
+ stat_directory.st_mode = F_file_mode_all_d | F_file_type_directory_d;
+ stat_regular.st_mode = F_file_mode_all_d | F_file_type_regular_d;
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
+ will_return(__wrap_f_file_stat, &stat_directory);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Target Directory pre-processing, begin, perform directory listing action.
+ will_return(__wrap_f_directory_empty, F_false);
+ will_return(__wrap_fl_directory_do, 2);
+
+ // The first child pre-processing.
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
+ will_return(__wrap_f_file_stat, &stat_regular);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // The second child pre-processing.
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_true); // A link, kt_remove_flag_file_operate_link_d is set.
+ will_return(__wrap_f_file_stat, &stat_regular);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Target Directory processing, begin.
+ will_return(__wrap_fl_directory_do, 3);
+
+ // The first child processing.
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, &child_1);
+ will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
+ will_return(__wrap_f_file_remove, F_okay);
+
+ // The second child processing.
+ will_return(__wrap_fl_directory_do, &child_2);
+ will_return(__wrap_fl_directory_do, &child_2);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_regular_e);
will_return(__wrap_f_file_remove, F_okay);
- // Parent Directory processing, continue.
- will_return(__wrap_fl_directory_do, &parent);
+ // Target Directory processing, continue.
+ will_return(__wrap_fl_directory_do, &target);
+ will_return(__wrap_fl_directory_do, &target);
will_return(__wrap_fl_directory_do, f_directory_recurse_do_flag_action_e | f_directory_recurse_do_flag_directory_e | f_directory_recurse_do_flag_top_after_e);
will_return(__wrap_f_directory_remove, F_okay);
*/
extern void test__kt_remove__directory_recurse_simple__one_child_one_exists_link(void **state);
+/**
+ * Test that the remove works for one file that exists and is not a link, for directory files that has one child with the recurse argument.
+ */
+extern void test__kt_remove__directory_recurse_simple__one_child_one_exists_link_not(void **state);
+
+/**
+ * Test that the remove works for one file that exists and is a link, for directory files that has two children with the recurse argument.
+ */
+extern void test__kt_remove__directory_recurse_simple__one_child_two_exists_link(void **state);
+
+/**
+ * Test that the remove works for one file that exists and is not a link, for directory files that has two children with the recurse argument.
+ */
+extern void test__kt_remove__directory_recurse_simple__one_child_two_exists_link_not(void **state);
+
#endif // _TEST__KT_remove__directory_recurse_simple
--- /dev/null
+#include "test-remove.h"
+#include "test-remove-directory_tree_simple.h"
+
+#include <program/kevux/tools/remove/remove/main.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+void test__kt_remove__directory_tree_simple__one_child_none_parent_one_exists_link(void **state) {
+
+ mock_unwrap = 0;
+
+ const f_string_static_t target_full = macro_f_string_static_t_initialize_1("parent_1/to_remove", 0, 18);
+
+ struct stat stat_directory;
+ struct stat stat_regular;
+
+ {
+ const f_string_t argv[] = { "mocked_main", target_full.string, "-t", 0 };
+
+ memset(&stat_directory, 0, sizeof(struct stat));
+ memset(&stat_regular, 0, sizeof(struct stat));
+
+ stat_directory.st_mode = F_file_mode_all_d | F_file_type_directory_d;
+ stat_regular.st_mode = F_file_mode_all_d | F_file_type_regular_d;
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_true); // A link, kt_remove_flag_file_operate_link_d is set.
+ will_return(__wrap_f_file_stat, &stat_regular);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Pre-process parent directory 1.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, F_false); // Not a link, kt_remove_flag_file_operate_link_d is not set.
+ will_return(__wrap_f_file_stat, &stat_directory);
+ will_return(__wrap_f_file_stat, F_okay);
+ will_return(__wrap_f_directory_empty, F_false);
+
+ // Target processing.
+ will_return(__wrap_f_file_remove, F_okay);
+
+ // Parent 1 processing.
+ will_return(__wrap_f_directory_remove, F_okay);
+
+ const int result = kt_main_test__remove(3, argv, 0);
+
+ assert_int_equal(result, 0);
+ }
+}
+
+#ifdef __cplusplus
+} // extern "C"
+#endif
--- /dev/null
+/**
+ * Kevux Tools - Remove
+ *
+ * Project: Kevux Tools
+ * API Version: 0.5
+ * Licenses: lgpl-2.1-or-later
+ *
+ * Test the remove.
+ */
+#ifndef _TEST__KT_remove__directory_tree_simple
+#define _TEST__KT_remove__directory_tree_simple
+
+/**
+ * Test that the remove works for one file that exists and is a link, for directory files that has no children and one parent with the tree argument.
+ */
+extern void test__kt_remove__directory_tree_simple__one_child_none_parent_one_exists_link(void **state);
+
+#endif // _TEST__KT_remove__directory_tree_simple
cmocka_unit_test(test__kt_remove__directory_no_args__two_empty_exists_not),
cmocka_unit_test(test__kt_remove__directory_recurse_simple__one_child_one_exists_link),
+ cmocka_unit_test(test__kt_remove__directory_recurse_simple__one_child_one_exists_link_not),
+ cmocka_unit_test(test__kt_remove__directory_recurse_simple__one_child_two_exists_link),
+ cmocka_unit_test(test__kt_remove__directory_recurse_simple__one_child_two_exists_link_not),
+
+ cmocka_unit_test(test__kt_remove__directory_tree_simple__one_child_none_parent_one_exists_link),
cmocka_unit_test(test__kt_remove__regular_no_args__one_exists_link),
cmocka_unit_test(test__kt_remove__regular_no_args__one_exists_link_not),
#include "test-remove-print_version.h"
#include "test-remove-directory_no_args.h"
#include "test-remove-directory_recurse_simple.h"
+#include "test-remove-directory_tree_simple.h"
#include "test-remove-regular_no_args.h"
#ifdef __cplusplus