I discovered that the `fl_directory_do()` could use some improvements while working on this.
Those changes are now made in the FLL project.
Make changes to this project based on the `fl_directory_do()` updates.
I began writing some tests and I encountered several problems and concerns that side-tracked me.
Rename the `kt_remove_operate_file_recurse()` to `kt_remove_operate_file_directory()`.
This should make its design and purpose more clear.
Make sure that the recurse status is processed in the proper order for the file operation directory.
Fix the problem where the recursion is being performed even when the recurse flag is not set.
Add checks to see if the directory is empty or not before deleting when recursion is not enabled.
The flag (called code in this case) needs to accurately reflect the directory status of the file during recursion.
Make sure the recurse error handler uses the top path or the recurse path as appropriate.
Print debug messages before recursion begins.
Add missing `f_status_t` return states to several print functions.
I did not do an exhaustive look (I imagine the TacocaT program needs similar changes).
Fix problems where the program output is being used for the print functions rather than the print output.
The runtime unit tests do not work correctly because the standard outputs are being closed.
Add additional ifdef checks to prevent the set down functions from being called.
build_libraries-level -lfll_2 -lfll_1 -lfll_0
build_libraries-monolithic -lfll
-build_sources_library common.c common/define.c common/enumeration.c common/print.c common/string.c common/type.c convert.c operate.c preprocess.c print/error.c print/message.c print/simulate.c print/verbose.c print/warning.c remove.c signal.c thread.c
+build_sources_library common.c common/define.c common/enumeration.c common/print.c common/string.c common/type.c convert.c operate.c preprocess.c print/debug.c print/error.c print/message.c print/simulate.c print/verbose.c print/warning.c remove.c signal.c thread.c
-build_sources_headers common.h common/define.h common/enumeration.h common/print.h common/string.h common/type.h convert.h operate.h preprocess.h print/error.h print/message.h print/simulate.h print/verbose.h print/warning.h remove.h signal.h thread.h
+build_sources_headers common.h common/define.h common/enumeration.h common/print.h common/string.h common/type.h convert.h operate.h preprocess.h print/debug.h print/error.h print/message.h print/simulate.h print/verbose.h print/warning.h remove.h signal.h thread.h
build_sources_documentation man
build_sources_headers remove.h main.h string.h
-build_objects_library common.o common/define.o common/enumeration.o common/print.o common/string.o common/type.o convert.o operate.o preprocess.o print/error.o print/message.o print/simulate.o print/verbose.o print/warning.o remove.o signal.o thread.o
+build_objects_library common.o common/define.o common/enumeration.o common/print.o common/string.o common/type.o convert.o operate.o preprocess.o print/debug.o print/error.o print/message.o print/simulate.o print/verbose.o print/warning.o remove.o signal.o thread.o
build_script no
build_shared yes
flags_program-android -fPIE -Wl,-z,relro
# Inject mocks.
+flags -Wl,--wrap=f_directory_remove
flags -Wl,--wrap=f_file_exists
flags -Wl,--wrap=f_file_remove
flags -Wl,--wrap=fll_program_print_version
build_indexer_arguments rcs
build_language c
-build_sources_object common.c common/define.c common/enumeration.c common/print.c common/string.c common/type.c convert.c operate.c preprocess.c print/error.c print/message.c print/simulate.c print/verbose.c print/warning.c remove.c signal.c thread.c
+build_sources_object common.c common/define.c common/enumeration.c common/print.c common/string.c common/type.c convert.c operate.c preprocess.c print/debug.c print/error.c print/message.c print/simulate.c print/verbose.c print/warning.c remove.c signal.c thread.c
-build_sources_headers common.h common/define.h common/enumeration.h common/print.h common/string.h common/type.h convert.h operate.h preprocess.h print/error.h print/message.h print/simulate.h print/verbose.h print/warning.h remove.h signal.h thread.h
+build_sources_headers common.h common/define.h common/enumeration.h common/print.h common/string.h common/type.h convert.h operate.h preprocess.h print/debug.h print/error.h print/message.h print/simulate.h print/verbose.h print/warning.h remove.h signal.h thread.h
build_sources_documentation man
build_sources_program test-remove.c
build_sources_program test-remove-print_help.c test-remove-print_version.c
+build_sources_program test-remove-regular_no_args.c
build_script no
build_shared yes
Programs such as the standard bold:"Linux" code:"install" (from bold:"GNU Coreutils") has some seriously simple problems such as being incapable of properly installing files when the file:"/proc" file system is mounted.
The reason for the failure (be it a bug or otherwise) is unclear but the dependency on code:"/proc" being mounted is ridiculous.
- - Better prepare for future planned functionality such as improved accessibility and interaction.
+ - Be better prepared for future planned functionality such as improved accessibility and interaction.
This should help make the bold:"Turtle Kevux" distribution more ready in regards to the designers and developers future plans.
- A need to improve maintainability of basic toolchain dependencies.
- Provides more exposure to the FLL:"Featureless Linux Library".
This results in more testing through real-word use.
- - Makes the system more familiar.
+ - Makes the system more comfortable.
The developer behind both the FLL:"Featureless Linux Library" and this project can work better in a more comfortable environment.
See: file:"dependencies" for specific dependencies of this project.
const f_string_t kt_remove_f_a[] = {
"f_console_parameter_prioritize_right",
"f_console_parameter_process",
+ "f_directory_remove",
"f_file_mode_from_string",
"f_file_mode_to_mode",
"f_file_remove",
enum {
kt_remove_f_f_console_parameter_prioritize_right_e,
kt_remove_f_f_console_parameter_process_e,
+ kt_remove_f_f_directory_remove_e,
kt_remove_f_f_file_mode_from_string_e,
kt_remove_f_f_file_mode_to_mode_e,
kt_remove_f_f_file_remove_e,
if (F_status_is_error_not(main->setting.state.status) && !(main->setting.flag & kt_remove_main_flag_simulate_d) && !(flag_operate & kt_remove_flag_file_operate_processed_d)) {
main->setting.state.status = flag_operate & kt_remove_flag_file_operate_directory_d
- ? kt_remove_operate_file_recurse(main, path, flag_operate)
+ ? kt_remove_operate_file_directory(main, path, flag_operate)
: kt_remove_operate_file_remove(main, path, flag_operate);
}
}
#endif // _di_kt_remove_operate_file_
-#ifndef _di_kt_remove_operate_file_recurse_
- f_status_t kt_remove_operate_file_recurse(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate) {
+#ifndef _di_kt_remove_operate_file_directory_
+ f_status_t kt_remove_operate_file_directory(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate) {
if (!kt_remove_operate_shall_remove(flag_operate)) return (flag_operate & kt_remove_flag_file_operate_remove_fail_d) ? F_status_set_error(F_no) : F_no;
recurse.state.code = flag_operate;
recurse.state.interrupt = &kt_remove_signal_check_recurse;
- recurse.flag = f_directory_recurse_do_flag_list_e;
- recurse.depth_max = kt_remove_depth_max_d;
+ recurse.flag = f_directory_recurse_do_flag_list_e | f_directory_recurse_do_flag_top_after_e;
+ recurse.depth_max = main->setting.flag & kt_remove_main_flag_recurse_d ? kt_remove_depth_max_d : 0;
- recurse.action = &kt_remove_operate_file_recurse_action;
+ recurse.action = &kt_remove_operate_file_directory_action;
recurse.handle = &kt_remove_operate_file_recurse_handle;
fl_directory_do(path, &recurse);
const f_status_t status = f_directory_recurse_do_delete(&recurse);
+ if (F_status_is_error(recurse.state.status)) return recurse.state.status;
+ if (F_status_is_error(status)) return status;
- return F_status_is_error(recurse.state.status)
- ? recurse.state.status
- : F_status_is_error(status)
- ? status
- : F_yes;
+ return F_yes;
}
-#endif // _di_kt_remove_operate_file_recurse_
+#endif // _di_kt_remove_operate_file_directory_
-#ifndef _di_kt_remove_operate_file_recurse_action_
- void kt_remove_operate_file_recurse_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint16_t flag) {
+#ifndef _di_kt_remove_operate_file_directory_action_
+ void kt_remove_operate_file_directory_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint32_t flag) {
if (!recurse || !recurse->state.custom || F_status_set_fine(recurse->state.status) == F_interrupt) return;
if (!kt_remove_operate_shall_remove(recurse->state.code) || !(flag & f_directory_recurse_do_flag_action_e)) return;
- recurse->state.status = kt_remove_operate_file_remove((kt_remove_main_t *) recurse->state.custom, recurse->path, recurse->state.code);
+ const f_string_static_t path = flag & f_directory_recurse_do_flag_top_after_e ? *recurse->path_top : recurse->path;
+
+ recurse->state.status = kt_remove_operate_file_remove(
+ (kt_remove_main_t *) recurse->state.custom,
+ path,
+ f_directory_is(path) == F_true
+ ? recurse->state.code | f_directory_recurse_do_flag_directory_e
+ : recurse->state.code & ~f_directory_recurse_do_flag_directory_e
+ );
}
-#endif // _di_kt_remove_operate_file_recurse_action_
+#endif // _di_kt_remove_operate_file_directory_action_
#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 uint16_t flag) {
+ void kt_remove_operate_file_recurse_handle(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint32_t flag) {
if (!recurse || !recurse->state.custom || F_status_set_fine(recurse->state.status) == F_interrupt) return;
return;
}
+ if (flag & f_directory_recurse_do_flag_list_e && recurse->state.status == F_status_set_error(F_recurse)) {
+ recurse->state.status = F_status_set_error(F_directory_empty_not);
+
+ kt_remove_print_error_file_status(&main->program.error, macro_kt_remove_f(fl_recurse_do), flag & f_directory_recurse_do_flag_top_after_e ? *recurse->path_top : recurse->path, f_file_operation_delete_s, fll_error_file_type_directory_e, recurse->state.status);
+
+ return;
+ }
+
// The top-level path is an empty string or an error occurred while processing the top-level path.
if (flag == (f_directory_recurse_do_flag_top_e | f_directory_recurse_do_flag_path_e) || flag == (f_directory_recurse_do_flag_top_e | f_directory_recurse_do_flag_path_e | f_directory_recurse_do_flag_before_e)) {
- kt_remove_print_error_file_status(&main->program.error, macro_kt_remove_f(fl_recurse_do), name, f_file_operation_stat_s, fll_error_file_type_path_e, recurse->state.status);
+ kt_remove_print_error_file_status(&main->program.error, macro_kt_remove_f(fl_recurse_do), flag & f_directory_recurse_do_flag_top_after_e ? *recurse->path_top : recurse->path, f_file_operation_stat_s, fll_error_file_type_path_e, recurse->state.status);
return;
}
// An error happened during directory list loading.
if (flag == (f_directory_recurse_do_flag_list_e | f_directory_recurse_do_flag_path_e)) {
- kt_remove_print_error_file_status(&main->program.error, macro_kt_remove_f(fl_recurse_do), name, f_file_operation_list_s, fll_error_file_type_directory_e, recurse->state.status);
+ kt_remove_print_error_file_status(&main->program.error, macro_kt_remove_f(fl_recurse_do), flag & f_directory_recurse_do_flag_top_after_e ? *recurse->path_top : recurse->path, f_file_operation_list_s, fll_error_file_type_directory_e, recurse->state.status);
return;
}
if (!main) return F_status_set_error(F_parameter);
if (!kt_remove_operate_shall_remove(flag_operate)) return (flag_operate & kt_remove_flag_file_operate_remove_fail_d) ? F_status_set_error(F_no) : F_no;
+ kt_remove_print_debug_operate_file_remove(&main->program.output, path, flag_operate);
+
f_status_t status = F_no;
if (flag_operate & kt_remove_flag_file_operate_follow_d) {
}
}
- status = (flag_operate & f_directory_recurse_do_flag_directory_e)
+ status = flag_operate & f_directory_recurse_do_flag_directory_e
? f_directory_remove((flag_operate & kt_remove_flag_file_operate_follow_d) ? main->cache.buffer : path, 0, F_false)
: f_file_remove((flag_operate & kt_remove_flag_file_operate_follow_d) ? main->cache.buffer : path);
if (F_status_is_error(status)) {
- kt_remove_print_error_file(&main->program.error, macro_kt_remove_f(f_file_remove), (flag_operate & kt_remove_flag_file_operate_follow_d) ? main->cache.buffer : path, f_file_operation_delete_s, fll_error_file_type_file_e);
+ kt_remove_print_error_file_status(
+ &main->program.error,
+ flag_operate & f_directory_recurse_do_flag_directory_e
+ ? macro_kt_remove_f(f_directory_remove)
+ : macro_kt_remove_f(f_file_remove),
+ flag_operate & kt_remove_flag_file_operate_follow_d
+ ? main->cache.buffer
+ : path,
+ f_file_operation_delete_s,
+ flag_operate & f_directory_recurse_do_flag_directory_e
+ ? fll_error_file_type_directory_e
+ : fll_error_file_type_file_e,
+ status
+ );
}
else {
+ kt_remove_print_verbose_operate_file_remove(&main->program.output, path, flag_operate);
+
status = F_yes;
}
*
* Errors (with error bit) from: f_string_dynamic_append().
*
- * Errors (with error bit) from: kt_remove_operate_file_recurse().
+ * 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().
*
* @see f_string_dynamic_append()
*
- * @see kt_remove_operate_file_recurse()
+ * @see kt_remove_operate_file_directory()
* @see kt_remove_operate_file_remove_delete()
* @see kt_remove_operate_file_remove()
* @see kt_remove_preprocess_file()
* @see f_file_remove()
* @see fl_directory_do()
*/
-#ifndef _di_kt_remove_operate_file_recurse_
- extern f_status_t kt_remove_operate_file_recurse(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate);
-#endif // _di_kt_remove_operate_file_recurse_
+#ifndef _di_kt_remove_operate_file_directory_
+ extern f_status_t kt_remove_operate_file_directory(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate);
+#endif // _di_kt_remove_operate_file_directory_
/**
* Perform directory recurse for a single file operation action.
* @see f_directory_remove()
* @see fl_directory_do()
*/
-#ifndef _di_kt_remove_operate_file_recurse_action_
- extern void kt_remove_operate_file_recurse_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint16_t flag);
-#endif // _di_kt_remove_operate_file_recurse_action_
+#ifndef _di_kt_remove_operate_file_directory_action_
+ extern void kt_remove_operate_file_directory_action(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_
/**
* Handle errors while performing directory recurse for a single file operation action.
* @see fl_directory_do()
*/
#ifndef _di_kt_remove_operate_file_recurse_handle_
- extern void kt_remove_operate_file_recurse_handle(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint16_t flag);
+ extern 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_recurse_handle_
/**
recurse.state.code = flag_operate;
recurse.state.interrupt = &kt_remove_signal_check_recurse;
- recurse.flag = f_directory_recurse_do_flag_list_e;
- recurse.depth_max = kt_remove_depth_max_d;
+ recurse.flag = f_directory_recurse_do_flag_list_e | f_directory_recurse_do_flag_top_after_e;
+ recurse.depth_max = main->setting.flag & kt_remove_main_flag_recurse_d ? kt_remove_depth_max_d : 0;
recurse.action = &kt_remove_preprocess_file_recurse_action;
recurse.handle = &kt_remove_operate_file_recurse_handle;
#endif // _di_kt_remove_preprocess_file_recurse_
#ifndef _di_kt_remove_preprocess_file_recurse_action_
- void kt_remove_preprocess_file_recurse_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint16_t flag) {
+ void kt_remove_preprocess_file_recurse_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint32_t flag) {
if (!recurse || !recurse->state.custom || F_status_set_fine(recurse->state.status) == F_interrupt) return;
* @see fl_directory_do()
*/
#ifndef _di_kt_remove_preprocess_file_recurse_action_
- extern void kt_remove_preprocess_file_recurse_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint16_t flag);
+ extern void kt_remove_preprocess_file_recurse_action(f_directory_recurse_do_t * const recurse, const f_string_static_t name, const uint32_t flag);
#endif // _di_kt_remove_preprocess_file_recurse_action_
#ifdef __cplusplus
--- /dev/null
+#include "../remove.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifndef _di_kt_remove_print_debug_operate_file_remove_
+ f_status_t kt_remove_print_debug_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint16_t flag) {
+
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
+
+ kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
+
+ if (main->setting.flag & kt_remove_main_flag_simulate_d) return F_output_not;
+ if (print->verbosity < f_console_verbosity_debug_e && !(main->setting.flag & kt_remove_main_flag_force_d)) return F_output_not;
+
+ fll_print_format("Removing '%Q'.%r", print->to, path, f_string_eol_s);
+
+ return F_okay;
+ }
+#endif // _di_kt_remove_print_debug_operate_file_remove_
+
+#ifdef __cplusplus
+} // extern "C"
+#endif
--- /dev/null
+/**
+ * Kevux Tools - Remove
+ *
+ * Project: Kevux Tools
+ * API Version: 0.5
+ * Licenses: lgpl-2.1-or-later
+ *
+ * Provides the print debug functionality.
+ *
+ * This is auto-included and should not need to be explicitly included.
+ */
+#ifndef _kt_remove_print_debug_h
+#define _kt_remove_print_debug_h
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * Print file removal message when debug.
+ *
+ * @param print
+ * The output structure to print to.
+ *
+ * This locks, uses, and unlocks the file stream.
+ *
+ * Must not be NULL.
+ *
+ * This does not alter print.custom.setting.state.status.
+ * @param path
+ * The path to the file to operate on.
+ * @param flag
+ * The file operate flags associated with the file.
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
+ */
+#ifndef _di_kt_remove_print_debug_operate_file_remove_
+ extern f_status_t kt_remove_print_debug_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint16_t flag);
+#endif // _di_kt_remove_print_debug_operate_file_remove_
+
+#ifdef __cplusplus
+} // extern "C"
+#endif
+
+#endif // _kt_remove_print_debug_h
#endif
#ifndef _di_kt_remove_print_simulate_operate_
- void kt_remove_print_simulate_operate(fl_print_t * const print) {
+ f_status_t kt_remove_print_simulate_operate(fl_print_t * const print) {
- if (!print || !print->custom) return;
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
- if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return F_output_not;
main->setting.state.status = F_okay;
f_print_dynamic(f_string_eol_s, print->to);
- f_file_stream_unlock(main->program.output.to);
+ f_file_stream_unlock(print->to);
+
+ return F_okay;
}
#endif // _di_kt_remove_print_simulate_operate_
#ifndef _di_kt_remove_print_simulate_operate_boolean_
- void kt_remove_print_simulate_operate_boolean(fl_print_t * const print, const f_string_static_t name, const bool yes) {
+ f_status_t kt_remove_print_simulate_operate_boolean(fl_print_t * const print, const f_string_static_t name, const bool yes) {
- if (!print || !print->custom) return;
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
- if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return F_output_not;
+
+ fll_print_format(" %r %r%r", print->to, name, yes ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
- fll_print_format(" %r %r%r", main->program.output.to, name, yes ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ return F_okay;
}
#endif // _di_kt_remove_print_simulate_operate_boolean_
#ifndef _di_kt_remove_print_simulate_operate_file_
- void kt_remove_print_simulate_operate_file(fl_print_t * const print, const f_string_static_t path, const uint16_t flag) {
+ f_status_t kt_remove_print_simulate_operate_file(fl_print_t * const print, const f_string_static_t path, const uint16_t flag) {
- if (!print || !print->custom) return;
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
- if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return F_output_not;
fll_print_format(
"%r%[Remove '%Q'%]:%r",
- main->program.output.to,
+ print->to,
flag & kt_remove_flag_file_operate_child_d
? f_string_eol_s
: f_string_empty_s,
- main->program.output.set->notable,
+ print->set->notable,
path,
- main->program.output.set->notable,
+ print->set->notable,
f_string_eol_s
);
+
+ return F_okay;
}
#endif // _di_kt_remove_print_simulate_operate_file_
#ifndef _di_kt_remove_print_simulate_operate_file_exists_
- void kt_remove_print_simulate_operate_file_exists(fl_print_t * const print, const f_string_static_t path, const uint16_t flag) {
+ f_status_t kt_remove_print_simulate_operate_file_exists(fl_print_t * const print, const f_string_static_t path, const uint16_t flag) {
- if (!print || !print->custom) return;
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
- if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return F_output_not;
if (F_status_is_error(main->setting.state.status)) {
- fl_print_format(" file_access_failure %ui%r", main->program.output.to, F_status_set_fine(main->setting.state.status), f_string_eol_s);
+ fl_print_format(" file_access_failure %ui%r", print->to, F_status_set_fine(main->setting.state.status), f_string_eol_s);
- return;
+ return F_status_set_error(F_output_not);
}
if (flag & kt_remove_flag_file_operate_link_d) {
const f_status_t status = f_file_link_read(path, F_false, &main->cache.buffer);
if (F_status_is_error(status)) {
- fl_print_format(" link_read_failure %ui%r", main->program.output.to, F_status_set_fine(status), f_string_eol_s);
+ fl_print_format(" link_read_failure %ui%r", print->to, F_status_set_fine(status), f_string_eol_s);
- return;
+ return F_status_set_error(F_output_not);
}
- f_file_stream_lock(main->program.output.to);
+ f_file_stream_lock(print->to);
- fl_print_format(" follow %r%r", main->program.output.to, (main->setting.flag & kt_remove_main_flag_follow_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
- fl_print_format(" to '%Q'%r", main->program.output.to, main->cache.buffer, f_string_eol_s);
+ fl_print_format(" follow %r%r", print->to, (main->setting.flag & kt_remove_main_flag_follow_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fl_print_format(" to '%Q'%r", print->to, main->cache.buffer, f_string_eol_s);
- f_file_stream_unlock(main->program.output.to);
+ f_file_stream_unlock(print->to);
}
- fll_print_format(" exists %r%r", main->program.output.to, main->setting.state.status == F_true ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" exists %r%r", print->to, main->setting.state.status == F_true ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+
+ return F_okay;
}
#endif // _di_kt_remove_print_simulate_operate_file_exists_
#ifndef _di_kt_remove_print_simulate_operate_file_stat_
- void kt_remove_print_simulate_operate_file_stat(fl_print_t * const print, struct stat statistics) {
+ f_status_t kt_remove_print_simulate_operate_file_stat(fl_print_t * const print, struct stat statistics) {
- if (!print || !print->custom) return;
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
- if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return F_output_not;
if (main->setting.flag & kt_remove_main_flag_block_d) {
- fll_print_format(" block %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_block_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" block %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_block_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_character_d) {
- fll_print_format(" character %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_character_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" character %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_character_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_directory_d) {
- fll_print_format(" directory %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_directory_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" directory %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_directory_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_fifo_d) {
- fll_print_format(" fifo %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_fifo_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" fifo %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_fifo_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_link_d) {
- fll_print_format(" link %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_link_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" link %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_link_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_regular_d) {
- fll_print_format(" regular %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_regular_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" regular %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_regular_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_socket_d) {
- fll_print_format(" socket %r%r", main->program.output.to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_socket_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" socket %r%r", print->to, (macro_f_file_type_get(statistics.st_mode) == F_file_type_socket_d) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
f_number_unsigned_t i = 0;
} // for
if (i == main->setting.users.used) {
- fll_print_format(" user%r", main->program.output.to, f_string_eol_s);
+ fll_print_format(" user%r", print->to, f_string_eol_s);
}
else {
- fll_print_format(" user %un%r", main->program.output.to, (f_number_unsigned_t) statistics.st_uid, f_string_eol_s);
+ fll_print_format(" user %un%r", print->to, (f_number_unsigned_t) statistics.st_uid, f_string_eol_s);
}
}
if (main->setting.flag & kt_remove_main_flag_same_d) {
- fll_print_format(" same %r%r", main->program.output.to, (statistics.st_uid == geteuid()) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
+ fll_print_format(" same %r%r", print->to, (statistics.st_uid == geteuid()) ? kt_remove_yes_s : kt_remove_no_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_different_d) {
- fll_print_format(" different %r%r", main->program.output.to, (statistics.st_uid == geteuid()) ? kt_remove_no_s : kt_remove_yes_s, f_string_eol_s);
+ fll_print_format(" different %r%r", print->to, (statistics.st_uid == geteuid()) ? kt_remove_no_s : kt_remove_yes_s, f_string_eol_s);
}
if (main->setting.flag & kt_remove_main_flag_group_d) {
} // for
if (i == main->setting.groups.used) {
- fll_print_format(" group%r", main->program.output.to, f_string_eol_s);
+ fll_print_format(" group%r", print->to, f_string_eol_s);
}
else {
- fll_print_format(" group %un%r", main->program.output.to, (f_number_unsigned_t) statistics.st_gid, f_string_eol_s);
+ fll_print_format(" group %un%r", print->to, (f_number_unsigned_t) statistics.st_gid, f_string_eol_s);
}
}
if (main->setting.flag & kt_remove_main_flag_mode_d) {
const mode_t mode = statistics.st_mode & F_file_mode_all_d;
- fll_print_format(" mode %@03un%r", main->program.output.to, (f_number_unsigned_t) mode, f_string_eol_s);
+ fll_print_format(" mode %@03un%r", print->to, (f_number_unsigned_t) mode, f_string_eol_s);
for (i = 0; i < main->setting.modes.used; ++i) {
for (j = 0; j < 4; ++j) {
if (main->setting.modes.array[i].type == types[j]) {
- fll_print_format(" mode_matched %Q %@03un%r", main->program.output.to, strings[j], (f_number_unsigned_t) main->setting.modes.array[i].mode, f_string_eol_s);
+ fll_print_format(" mode_matched %Q %@03un%r", print->to, strings[j], (f_number_unsigned_t) main->setting.modes.array[i].mode, f_string_eol_s);
break;
}
for (i = 0; i < dates[j]->used; ++i) {
- if (kt_remove_signal_check(main)) return;
+ if (kt_remove_signal_check(main)) return F_status_set_error(F_interrupt);
match_year = kt_remove_time_year_unix_epoch_d + (times[j].tv_sec / kt_remove_time_seconds_in_year_d);
match_second = times[j].tv_sec % kt_remove_time_seconds_in_year_d;
}
if (name_type.used) {
- fll_print_format(" %Q %Q ", main->program.output.to, *names[j], result ? kt_remove_yes_s : kt_remove_no_s);
- fll_print_format("%u::%un 0:%un %Q ", main->program.output.to, match_year, (f_number_unsigned_t) times[j].tv_sec, (f_number_unsigned_t) times[j].tv_nsec, name_type);
- fll_print_format("%u::%un 0:%un%r", main->program.output.to, dates[j]->array[i].start_year, dates[j]->array[i].start_second, dates[j]->array[i].start_nanosecond, f_string_eol_s);
+ fll_print_format(" %Q %Q ", print->to, *names[j], result ? kt_remove_yes_s : kt_remove_no_s);
+ fll_print_format("%u::%un 0:%un %Q ", print->to, match_year, (f_number_unsigned_t) times[j].tv_sec, (f_number_unsigned_t) times[j].tv_nsec, name_type);
+ fll_print_format("%u::%un 0:%un%r", print->to, dates[j]->array[i].start_year, dates[j]->array[i].start_second, dates[j]->array[i].start_nanosecond, f_string_eol_s);
break;
}
} // for
} // for
}
+
+ return F_okay;
}
#endif // _di_kt_remove_print_simulate_operate_file_stat_
#ifndef _di_kt_remove_print_simulate_operate_prompt_once_
- void kt_remove_print_simulate_operate_prompt_once(fl_print_t * const print, const bool yes) {
+ f_status_t kt_remove_print_simulate_operate_prompt_once(fl_print_t * const print, const bool yes) {
- if (!print || !print->custom) return;
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
- if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return;
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_d)) return F_output_not;
+
+ fll_print_format(" %r %r%r", print->to, kt_remove_prompt_s, yes ? kt_remove_once_s : kt_remove_no_s, f_string_eol_s);
- fll_print_format(" %r %r%r", main->program.output.to, kt_remove_prompt_s, yes ? kt_remove_once_s : kt_remove_no_s, f_string_eol_s);
+ return F_okay;
}
#endif // _di_kt_remove_print_simulate_operate_prompt_once_
* Must not be NULL.
*
* This does not alter print.custom.setting.state.status.
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
*/
#ifndef _di_kt_remove_print_simulate_operate_
- extern void kt_remove_print_simulate_operate(fl_print_t * const print);
+ extern f_status_t kt_remove_print_simulate_operate(fl_print_t * const print);
#endif // _di_kt_remove_print_simulate_operate_
/**
* @param yes
* If TRUE, then print "yes".
* If FALSE, then print "no".
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
*/
#ifndef _di_kt_remove_print_simulate_operate_boolean_
- extern void kt_remove_print_simulate_operate_boolean(fl_print_t * const print, const f_string_static_t name, const bool yes);
+ extern f_status_t kt_remove_print_simulate_operate_boolean(fl_print_t * const print, const f_string_static_t name, const bool yes);
#endif // _di_kt_remove_print_simulate_operate_boolean_
/**
* @param flag
* The file operate flags associated with the file.
* The kt_remove_flag_file_operate_link_d is of specific interest here.
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
*/
#ifndef _di_kt_remove_print_simulate_operate_file_
- extern void kt_remove_print_simulate_operate_file(fl_print_t * const print, const f_string_static_t path, const uint16_t flag);
+ extern f_status_t kt_remove_print_simulate_operate_file(fl_print_t * const print, const f_string_static_t path, const uint16_t flag);
#endif // _di_kt_remove_print_simulate_operate_file_
/**
* @param flag
* The file operate flags associated with the file.
* The kt_remove_flag_file_operate_link_d is of specific interest here.
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
*/
#ifndef _di_kt_remove_print_simulate_operate_file_exists_
- extern void kt_remove_print_simulate_operate_file_exists(fl_print_t * const print, const f_string_static_t path, const uint16_t flag);
+ extern f_status_t kt_remove_print_simulate_operate_file_exists(fl_print_t * const print, const f_string_static_t path, const uint16_t flag);
#endif // _di_kt_remove_print_simulate_operate_file_exists_
/**
* This does not alter print.custom.setting.state.status.
* @param statistics
* The file statistics.
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
*/
#ifndef _di_kt_remove_print_simulate_operate_file_stat_
- extern void kt_remove_print_simulate_operate_file_stat(fl_print_t * const print, struct stat statistics);
+ extern f_status_t kt_remove_print_simulate_operate_file_stat(fl_print_t * const print, struct stat statistics);
#endif // _di_kt_remove_print_simulate_operate_file_stat_
/**
* @param yes
* If TRUE, then print "once".
* If FALSE, then print "no".
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
*/
#ifndef _di_kt_remove_print_simulate_operate_prompt_once_
- extern void kt_remove_print_simulate_operate_prompt_once(fl_print_t * const print, const bool yes);
+ extern f_status_t kt_remove_print_simulate_operate_prompt_once(fl_print_t * const print, const bool yes);
#endif // _di_kt_remove_print_simulate_operate_prompt_once_
#ifdef __cplusplus
extern "C" {
#endif
+#ifndef _di_kt_remove_print_verbose_operate_file_remove_
+ f_status_t kt_remove_print_verbose_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint16_t flag) {
+
+ if (!print || !print->custom) return F_status_set_error(F_output_not);
+
+ kt_remove_main_t * const main = (kt_remove_main_t *) print->custom;
+
+ if (main->setting.flag & kt_remove_main_flag_simulate_d) return F_output_not;
+ if (print->verbosity < f_console_verbosity_verbose_e && !(main->setting.flag & kt_remove_main_flag_force_d)) return F_output_not;
+
+ fll_print_format("Removed '%Q'.%r", print->to, path, f_string_eol_s);
+
+ return F_okay;
+ }
+#endif // _di_kt_remove_print_verbose_operate_file_remove_
+
#ifdef __cplusplus
} // extern "C"
#endif
extern "C" {
#endif
+/**
+ * Print file removal message when verbose.
+ *
+ * @param print
+ * The output structure to print to.
+ *
+ * This locks, uses, and unlocks the file stream.
+ *
+ * Must not be NULL.
+ *
+ * This does not alter print.custom.setting.state.status.
+ * @param path
+ * The path to the file to operate on.
+ * @param flag
+ * The file operate flags associated with the file.
+ *
+ * @return
+ * F_okay on success.
+ * F_output_not on success, but no printing is performed.
+ *
+ * F_output_not (with error bit) if setting is NULL.
+ */
+#ifndef _di_kt_remove_print_verbose_operate_file_remove_
+ extern f_status_t kt_remove_print_verbose_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint16_t flag);
+#endif // _di_kt_remove_print_verbose_operate_file_remove_
+
#ifdef __cplusplus
} // extern "C"
#endif
#include <program/kevux/tools/remove/main/convert.h>
#include <program/kevux/tools/remove/main/operate.h>
#include <program/kevux/tools/remove/main/preprocess.h>
+#include <program/kevux/tools/remove/main/print/debug.h>
#include <program/kevux/tools/remove/main/print/error.h>
#include <program/kevux/tools/remove/main/print/message.h>
#include <program/kevux/tools/remove/main/print/simulate.h>
kt_remove_main_delete(&data);
- fll_program_standard_set_down(&data.program);
+ // Must not close the standard I/O during testing.
+ #ifndef _kt_MOCK_MAIN_NAME
+ fll_program_standard_set_down(&data.program);
+ #endif // _kt_MOCK_MAIN_NAME
return (F_status_is_error(data.setting.state.status) || data.setting.state.status == F_false) ? 1 : 0;
}
int mock_unwrap = 0;
f_status_t mock_status = 0;
+f_status_t __wrap_f_directory_remove(const f_string_static_t path, const int depth_max, const bool preserve) {
+
+ const bool failure = mock_type(bool);
+
+ if (failure) return mock_type(f_status_t);
+
+ return mock_type(f_status_t);
+}
+
f_status_t __wrap_f_file_exists(const f_string_static_t path, const bool dereference) {
const bool failure = mock_type(bool);
extern int mock_unwrap;
extern f_status_t mock_status;
+extern f_status_t __wrap_f_directory_remove(const f_string_static_t path, const int depth_max, const bool preserve);
+
extern f_status_t __wrap_f_file_exists(const f_string_static_t path, const bool dereference);
extern f_status_t __wrap_f_file_remove(const f_string_static_t path);
* API Version: 0.5
* Licenses: lgpl-2.1-or-later
*
- * Test the --help parameter.
+ * Test the --version parameter.
*/
#ifndef _TEST__KT_remove__print_version
#define _TEST__KT_remove__print_version
--- /dev/null
+#include "test-remove.h"
+#include "test-remove-regular_no_args.h"
+
+#include <program/kevux/tools/remove/remove/main.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+void test__kt_remove__regular_no_args__works(void **state) {
+
+ mock_unwrap = 0;
+
+ {
+ const f_string_t argv[] = { "mocked_main", "to_remove", 0 };
+
+ mock_status = F_okay;
+
+ will_return(__wrap_f_file_exists, F_false);
+ will_return(__wrap_f_file_exists, F_false);
+
+ // @todo this is incomplete.
+ //will_return(__wrap_f_file_remove, F_test);
+
+ const int result = kt_main_test__remove(2, argv, 0);
+
+ assert_int_equal(result, 0);
+ assert_int_equal(mock_status, F_okay);
+ }
+}
+
+#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__regular_no_args
+#define _TEST__KT_remove__regular_no_args
+
+/**
+ * Test that the remove works, for regular files without other arguments.
+ */
+extern void test__kt_remove__regular_no_args__works(void **state);
+
+#endif // _TEST__KT_remove__regular_no_args
const struct CMUnitTest tests[] = {
cmocka_unit_test(test__kt_remove__print_help__works),
cmocka_unit_test(test__kt_remove__print_version__works),
+
+ cmocka_unit_test(test__kt_remove__regular_no_args__works),
};
return cmocka_run_group_tests(tests, setup, setdown);
// Test includes.
#include "test-remove-print_help.h"
#include "test-remove-print_version.h"
+#include "test-remove-regular_no_args.h"
#ifdef __cplusplus
extern "C" {