From ff8def1c6a94c6d80718e9d25c1f42fd1c23e662 Mon Sep 17 00:00:00 2001 From: Kevin Day Date: Tue, 25 Mar 2025 19:24:36 -0500 Subject: [PATCH] Progress: Continue working on completing the remove program. Add UNIX Timestamp tests. Fix date changed tests where I forgot to reset temporary date loop variable assignment before committing. Add the accessed and updated (modified) time tests. When recursion is disabled and a directory is being followed, then prevent a double delete by not performing the delete on the "after" at the top level. Implement the `--prompt` support. Make sure to take in command line input. Make sure to quit the program when prompt is one time and the answer is no. Change the flag to 32-bit for consistency reasons. --- data/build/remove/settings-tests.remove | 3 +- data/build/stand_alone/configs/remove-config.h | 9 +- data/build/stand_alone/configs/tacocat-config.h | 1 + .../program/kevux/tools/remove/main/common/print.c | 1 + .../program/kevux/tools/remove/main/common/print.h | 1 + .../program/kevux/tools/remove/main/common/type.c | 1 + .../program/kevux/tools/remove/main/common/type.h | 10 +- .../c/program/kevux/tools/remove/main/operate.c | 95 ++- .../c/program/kevux/tools/remove/main/operate.h | 31 +- .../c/program/kevux/tools/remove/main/preprocess.c | 8 +- .../c/program/kevux/tools/remove/main/preprocess.h | 6 +- .../program/kevux/tools/remove/main/print/debug.c | 2 +- .../program/kevux/tools/remove/main/print/debug.h | 2 +- .../kevux/tools/remove/main/print/message.c | 16 + .../kevux/tools/remove/main/print/message.h | 24 + .../kevux/tools/remove/main/print/simulate.c | 4 +- .../kevux/tools/remove/main/print/simulate.h | 4 +- .../kevux/tools/remove/main/print/verbose.c | 2 +- .../kevux/tools/remove/main/print/verbose.h | 2 +- sources/c/program/kevux/tools/remove/main/remove.c | 10 +- tests/unit/remove/c/test-remove-date_accessed.c | 780 +++++++++++++++++++++ tests/unit/remove/c/test-remove-date_accessed.h | 38 + tests/unit/remove/c/test-remove-date_changed.c | 2 +- tests/unit/remove/c/test-remove-date_updated.c | 780 +++++++++++++++++++++ tests/unit/remove/c/test-remove-date_updated.h | 38 + tests/unit/remove/c/test-remove-unix.c | 161 +++++ tests/unit/remove/c/test-remove-unix.h | 18 + tests/unit/remove/c/test-remove.c | 14 + tests/unit/remove/c/test-remove.h | 3 + 29 files changed, 2026 insertions(+), 40 deletions(-) create mode 100644 tests/unit/remove/c/test-remove-date_accessed.c create mode 100644 tests/unit/remove/c/test-remove-date_accessed.h create mode 100644 tests/unit/remove/c/test-remove-date_updated.c create mode 100644 tests/unit/remove/c/test-remove-date_updated.h create mode 100644 tests/unit/remove/c/test-remove-unix.c create mode 100644 tests/unit/remove/c/test-remove-unix.h diff --git a/data/build/remove/settings-tests.remove b/data/build/remove/settings-tests.remove index a2d7884..15047b5 100644 --- a/data/build/remove/settings-tests.remove +++ b/data/build/remove/settings-tests.remove @@ -32,11 +32,12 @@ build_libraries-individual_thread -lf_thread build_sources_program test-remove.c build_sources_program test-remove-print_help.c test-remove-print_version.c build_sources_program test-remove-file_mode.c test-remove-file_type.c -build_sources_program test-remove-date_changed.c +build_sources_program test-remove-date_accessed.c test-remove-date_changed.c test-remove-date_updated.c build_sources_program test-remove-directory_no_args.c test-remove-directory_recurse_simple.c test-remove-directory_tree_simple.c build_sources_program test-remove-force.c build_sources_program test-remove-group.c build_sources_program test-remove-regular_no_args.c +build_sources_program test-remove-unix.c build_sources_program test-remove-user.c build_script no diff --git a/data/build/stand_alone/configs/remove-config.h b/data/build/stand_alone/configs/remove-config.h index b3c8fa9..8d58a0f 100644 --- a/data/build/stand_alone/configs/remove-config.h +++ b/data/build/stand_alone/configs/remove-config.h @@ -385,7 +385,7 @@ #define _di_f_file_operation_open_s_ #define _di_f_file_operation_pipe_s_ //#define _di_f_file_operation_process_s_ -#define _di_f_file_operation_read_s_ +//#define _di_f_file_operation_read_s_ #define _di_f_file_operation_receive_s_ #define _di_f_file_operation_rename_s_ //#define _di_f_file_operation_s_ @@ -402,7 +402,7 @@ #define _di_f_file_owner_read_ #define _di_f_file_poll_ #define _di_f_file_read_ -#define _di_f_file_read_block_ +//#define _di_f_file_read_block_ #define _di_f_file_read_until_ //#define _di_f_file_remove_ #define _di_f_file_remove_at_ @@ -788,7 +788,7 @@ #define _di_f_string_ascii_V_s_ //#define _di_f_string_ascii_W_s_ //#define _di_f_string_ascii_X_s_ -#define _di_f_string_ascii_Y_s_ +//#define _di_f_string_ascii_Y_s_ //#define _di_f_string_ascii_Z_s_ //#define _di_f_string_ascii_a_s_ #define _di_f_string_ascii_acknowledge_negative_s_ @@ -880,7 +880,7 @@ #define _di_f_string_ascii_v_s_ //#define _di_f_string_ascii_w_s_ //#define _di_f_string_ascii_x_s_ -#define _di_f_string_ascii_y_s_ +//#define _di_f_string_ascii_y_s_ //#define _di_f_string_ascii_z_s_ #define _di_f_string_constant_t_ #define _di_f_string_dynamic_append_ @@ -1639,6 +1639,7 @@ //#define _di_fll_error_file_type_directory_s_ //#define _di_fll_error_file_type_e_ //#define _di_fll_error_file_type_file_s_ +//#define _di_fll_error_file_type_input_s_ //#define _di_fll_error_file_type_link_s_ //#define _di_fll_error_file_type_path_s_ //#define _di_fll_error_file_type_pipe_s_ diff --git a/data/build/stand_alone/configs/tacocat-config.h b/data/build/stand_alone/configs/tacocat-config.h index 7ca2147..28d111d 100644 --- a/data/build/stand_alone/configs/tacocat-config.h +++ b/data/build/stand_alone/configs/tacocat-config.h @@ -1867,6 +1867,7 @@ //#define _di_fll_error_file_type_directory_s_ //#define _di_fll_error_file_type_e_ //#define _di_fll_error_file_type_file_s_ +#define _di_fll_error_file_type_input_s_ //#define _di_fll_error_file_type_link_s_ //#define _di_fll_error_file_type_path_s_ //#define _di_fll_error_file_type_pipe_s_ diff --git a/sources/c/program/kevux/tools/remove/main/common/print.c b/sources/c/program/kevux/tools/remove/main/common/print.c index cfb8214..8e214d8 100644 --- a/sources/c/program/kevux/tools/remove/main/common/print.c +++ b/sources/c/program/kevux/tools/remove/main/common/print.c @@ -11,6 +11,7 @@ extern "C" { "f_directory_remove", "f_file_mode_from_string", "f_file_mode_to_mode", + "f_file_read_block", "f_file_remove", "f_memory_array_increase", "f_memory_array_increase_by", diff --git a/sources/c/program/kevux/tools/remove/main/common/print.h b/sources/c/program/kevux/tools/remove/main/common/print.h index 060027d..968edb6 100644 --- a/sources/c/program/kevux/tools/remove/main/common/print.h +++ b/sources/c/program/kevux/tools/remove/main/common/print.h @@ -44,6 +44,7 @@ extern "C" { 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_read_block_e, kt_remove_f_f_file_remove_e, kt_remove_f_f_memory_array_increase_e, kt_remove_f_f_memory_array_increase_by_e, diff --git a/sources/c/program/kevux/tools/remove/main/common/type.c b/sources/c/program/kevux/tools/remove/main/common/type.c index 8cb0fda..3a459bb 100644 --- a/sources/c/program/kevux/tools/remove/main/common/type.c +++ b/sources/c/program/kevux/tools/remove/main/common/type.c @@ -10,6 +10,7 @@ extern "C" { if (!cache) return; f_memory_array_resize(0, sizeof(f_char_t), (void **) &cache->buffer.string, &cache->buffer.used, &cache->buffer.size); + f_memory_array_resize(0, sizeof(f_char_t), (void **) &cache->input.string, &cache->input.used, &cache->input.size); f_memory_arrays_resize(0, sizeof(f_string_dynamic_t), (void **) &cache->files.array, &cache->files.used, &cache->files.size, &f_string_dynamics_delete_callback); f_memory_arrays_resize(0, sizeof(f_string_dynamic_t), (void **) &cache->memory.array, &cache->memory.used, &cache->memory.size, &f_string_dynamics_delete_callback); diff --git a/sources/c/program/kevux/tools/remove/main/common/type.h b/sources/c/program/kevux/tools/remove/main/common/type.h index 5096df6..d27a724 100644 --- a/sources/c/program/kevux/tools/remove/main/common/type.h +++ b/sources/c/program/kevux/tools/remove/main/common/type.h @@ -141,6 +141,7 @@ extern "C" { * The program cache. * * buffer: The generic buffer. + * input: The buffer used specifically for handling input. * files: A collection of files, often used during path recursion like those associated with the tree parameter. * memory: A list of paths or partial paths representing files already processed. * tree: A collection of files to process as a result of the --tree command. @@ -148,6 +149,7 @@ extern "C" { #ifndef _di_kt_remove_cache_t_ typedef struct { f_string_dynamic_t buffer; + f_string_dynamic_t input; f_string_dynamics_t files; f_string_dynamics_t memory; f_string_dynamics_t tree; @@ -156,6 +158,7 @@ extern "C" { #define kt_remove_cache_t_initialize \ { \ f_string_dynamic_t_initialize, \ + f_string_dynamic_t_initialize, \ f_string_dynamics_t_initialize, \ f_string_dynamics_t_initialize, \ f_string_dynamics_t_initialize, \ @@ -174,7 +177,7 @@ extern "C" { typedef f_status_t (*print_help_call_t)(fl_print_t * const print, const f_color_context_t context); typedef void (*process_normal_call_t)(kt_remove_main_t * const main); typedef void (*process_operate_file_call_t)(kt_remove_main_t * const main, const f_string_static_t path, const struct stat statistics, uint16_t * const flag); - typedef void (*process_operate_file_simulate_call_t)(kt_remove_main_t * const main, const f_string_static_t path, const struct stat statistics, const uint16_t flag_operate, uint16_t * const flag); + typedef void (*process_operate_file_simulate_call_t)(kt_remove_main_t * const main, const f_string_static_t path, const struct stat statistics, const uint32_t flag_operate, uint32_t * const flag); typedef struct { print_help_call_t print_help; @@ -198,7 +201,8 @@ extern "C" { * This is passed to the program-specific main entry point to designate program settings. * These program settings are often processed from the program arguments (often called the command line arguments). * - * flag: Flags passed to the main function. + * flag: Flags passed to the main function. + * prompt: The number of times the files have been prompted (generally will not be larger than 4). * * status_thread: A status used eclusively by the threaded signal handler. * state: The state data used when processing data. @@ -218,6 +222,7 @@ extern "C" { #ifndef _di_kt_remove_setting_t_ typedef struct { uint64_t flag; + uint8_t prompt; f_status_t status_thread; f_state_t state; @@ -238,6 +243,7 @@ extern "C" { #define kt_remove_setting_t_initialize \ { \ kt_remove_main_flag_none_d, \ + 0, \ F_okay, \ macro_f_state_t_initialize_1(kt_remove_allocation_large_d, kt_remove_allocation_small_d, F_okay, 0, 0, &fll_program_standard_signal_handle, 0, 0, 0, 0), \ f_string_dynamics_t_initialize, \ diff --git a/sources/c/program/kevux/tools/remove/main/operate.c b/sources/c/program/kevux/tools/remove/main/operate.c index 75ca8a1..4d1af67 100644 --- a/sources/c/program/kevux/tools/remove/main/operate.c +++ b/sources/c/program/kevux/tools/remove/main/operate.c @@ -17,13 +17,17 @@ extern "C" { if (kt_remove_signal_check(main)) return; - const uint16_t flag_operate = kt_remove_preprocess_file(main, path, 0); + const uint32_t flag_operate = kt_remove_preprocess_file(main, path, 0); f_status_t status = F_no; 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)) { - status = main->setting.state.status = flag_operate & kt_remove_flag_file_operate_directory_d - ? kt_remove_operate_file_directory(main, path, flag_operate) - : kt_remove_operate_file_remove(main, path, flag_operate); + kt_remove_operate_file_prompt(main, path, flag_operate); + + if (F_status_is_error_not(main->setting.state.status) && main->setting.state.status != F_skip) { + status = main->setting.state.status = flag_operate & kt_remove_flag_file_operate_directory_d + ? kt_remove_operate_file_directory(main, path, flag_operate) + : kt_remove_operate_file_remove(main, path, flag_operate); + } } if (F_status_is_error_not(main->setting.state.status)) { @@ -41,7 +45,7 @@ extern "C" { #endif // _di_kt_remove_operate_file_ #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) { + f_status_t kt_remove_operate_file_directory(kt_remove_main_t * const main, const f_string_static_t path, const uint32_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; @@ -53,7 +57,7 @@ extern "C" { recurse.state.interrupt = &kt_remove_signal_check_recurse; recurse.depth_max = main->setting.flag & kt_remove_main_flag_recurse_d ? kt_remove_depth_max_d : 0; - recurse.flag = f_directory_recurse_do_flag_top_after_e; + recurse.flag = main->setting.flag & kt_remove_main_flag_recurse_d ? f_directory_recurse_do_flag_top_after_e : 0; recurse.path_top = &path; recurse.action = &kt_remove_operate_file_directory_action; @@ -108,7 +112,7 @@ extern "C" { 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); + const uint32_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; @@ -130,6 +134,75 @@ extern "C" { } #endif // _di_kt_remove_operate_file_parent_ +#ifndef _di_kt_remove_operate_file_prompt_ + void kt_remove_operate_file_prompt(kt_remove_main_t * const main, const f_string_static_t path, const uint flag_operate) { + + if (!main) return; + + if (!(main->setting.flag & kt_remove_main_flag_prompt_all_d) || (main->setting.flag & kt_remove_main_flag_prompt_never_d)) { + main->setting.state.status = F_okay; + + return; + } + + if (!path.used) { + main->setting.state.status = F_data_not; + + return; + } + + if (main->setting.flag & kt_remove_main_flag_prompt_once_d) { + if (main->setting.prompt) return; + + main->setting.prompt = 1; + } + else if (main->setting.flag & kt_remove_main_flag_prompt_follow_d) { + if (!(flag_operate & kt_remove_flag_file_operate_link_d)) return; + } + + main->cache.input.used = 0; + + kt_remove_print_message_remove_confirm(&main->program.message, path); + + // @todo This does not work well with interrupts, so another f_file_read_block_interrupt() needs to be created in the FLL project (or add interrupt callback to each f_file_read function). + main->setting.state.status = f_file_read_block(main->program.input, &main->cache.input); + + if (F_status_is_error(main->setting.state.status)) { + kt_remove_print_error_file(&main->program.error, macro_kt_remove_f(f_file_read_block), f_string_empty_s, f_file_operation_read_s, fll_error_file_type_input_e); + + return; + } + + // Skip past any NULL characters in the input. + { + f_number_unsigned_t i = 0; + + for (; i < main->cache.input.used && !main->cache.input.string[i]; ++i) { + if (kt_remove_signal_check(main)) return; + } // for + + // Only 'y' and 'Y' are treated as confimation and all others equate to 'no'. + if (main->cache.input.string[i] == f_string_ascii_y_s.string[0] || main->cache.input.string[i] == f_string_ascii_Y_s.string[0]) { + main->setting.state.status = F_okay; + + // Shrink an oversized input buffer. + if (main->cache.input.size > kt_remove_allocation_large_d) { + f_memory_array_resize(0, sizeof(f_char_t), (void **) &main->cache.input.string, &main->cache.input.used, &main->cache.input.size); + } + + return; + } + } + + // Shrink an oversized input buffer. + if (main->cache.input.size > kt_remove_allocation_large_d) { + f_memory_array_resize(0, sizeof(f_char_t), (void **) &main->cache.input.string, &main->cache.input.used, &main->cache.input.size); + } + + main->setting.state.status = (main->setting.flag & kt_remove_main_flag_prompt_once_d) ? F_status_set_error(F_skip) : F_skip; + } +#endif // _di_kt_remove_operate_file_prompt_ + #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) { @@ -169,7 +242,7 @@ extern "C" { #endif // _di_kt_remove_operate_file_recurse_handle_ #ifndef _di_kt_remove_operate_file_remove_normal_ - f_status_t kt_remove_operate_file_remove(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate) { + f_status_t kt_remove_operate_file_remove(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate) { 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; @@ -221,7 +294,7 @@ extern "C" { #endif // _di_kt_remove_operate_file_remove_normal_ #ifndef _di_kt_remove_operate_memory_check_ - void kt_remove_operate_memory_check(kt_remove_main_t * const main, const f_string_static_t path, uint16_t * const flag_operate) { + void kt_remove_operate_memory_check(kt_remove_main_t * const main, const f_string_static_t path, uint32_t * const flag_operate) { if (!main) return; @@ -295,7 +368,7 @@ extern "C" { #endif // _di_kt_remove_operate_memory_check_ #ifndef _di_kt_remove_operate_memory_save_ - void kt_remove_operate_memory_save(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate) { + void kt_remove_operate_memory_save(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate) { if (!main) return; @@ -380,7 +453,7 @@ extern "C" { #endif // _di_kt_remove_operate_memory_save_ #ifndef _di_kt_remove_operate_shall_remove_ - f_status_t kt_remove_operate_shall_remove(const uint16_t flag) { + f_status_t kt_remove_operate_shall_remove(const uint32_t flag) { return (flag & kt_remove_flag_file_operate_remove_d) && !(flag & kt_remove_flag_file_operate_remove_not_fail_d); } diff --git a/sources/c/program/kevux/tools/remove/main/operate.h b/sources/c/program/kevux/tools/remove/main/operate.h index c7c5da2..4c54ad9 100644 --- a/sources/c/program/kevux/tools/remove/main/operate.h +++ b/sources/c/program/kevux/tools/remove/main/operate.h @@ -77,7 +77,7 @@ extern "C" { * @see fl_directory_do() */ #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); + extern f_status_t kt_remove_operate_file_directory(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate); #endif // _di_kt_remove_operate_file_directory_ /** @@ -134,6 +134,27 @@ extern "C" { #endif // _di_kt_remove_operate_file_parent_ /** + * Perform prompting, if needed, before removing file. + * + * @param main + * The main program and settings data. + * + * Must not be NULL. + * + * This alters main.setting.state.status: + * F_okay on success or if prompting is not requested. + * F_data_not on success but path is an empty string. + * F_skip on success and the file is not to be removed (skipped). + * + * F_skip (with error bit) if skipping should be applied to all paths to be removed from this point forward. + * @param path + * The path to the file to operate on. + */ +#ifndef _di_kt_remove_operate_file_prompt_ + extern void kt_remove_operate_file_prompt(kt_remove_main_t * const main, const f_string_static_t path, const uint flag_operate); +#endif // _di_kt_remove_operate_file_prompt_ + +/** * Handle errors while performing directory recurse for a single file operation action. * * @param recurse @@ -182,7 +203,7 @@ extern "C" { * @see f_file_remove() */ #ifndef _di_kt_remove_operate_file_remove_normal_ - extern f_status_t kt_remove_operate_file_remove(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate); + extern f_status_t kt_remove_operate_file_remove(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate); #endif // _di_kt_remove_operate_file_remove_normal_ /** @@ -210,7 +231,7 @@ extern "C" { * @see f_string_dynamic_append() */ #ifndef _di_kt_remove_operate_memory_check_ - extern void kt_remove_operate_memory_check(kt_remove_main_t * const main, const f_string_static_t path, uint16_t * const flag_operate); + extern void kt_remove_operate_memory_check(kt_remove_main_t * const main, const f_string_static_t path, uint32_t * const flag_operate); #endif // _di_kt_remove_operate_memory_check_ /** @@ -236,7 +257,7 @@ extern "C" { * @see f_string_dynamic_append() */ #ifndef _di_kt_remove_operate_memory_save_ - extern void kt_remove_operate_memory_save(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate); + extern void kt_remove_operate_memory_save(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate); #endif // _di_kt_remove_operate_memory_save_ /** @@ -250,7 +271,7 @@ extern "C" { * F_false otherwise. */ #ifndef _di_kt_remove_operate_shall_remove_ - extern f_status_t kt_remove_operate_shall_remove(const uint16_t flag); + extern f_status_t kt_remove_operate_shall_remove(const uint32_t flag); #endif // _di_kt_remove_operate_shall_remove_ #ifdef __cplusplus diff --git a/sources/c/program/kevux/tools/remove/main/preprocess.c b/sources/c/program/kevux/tools/remove/main/preprocess.c index 7f2dc5d..3b071ee 100644 --- a/sources/c/program/kevux/tools/remove/main/preprocess.c +++ b/sources/c/program/kevux/tools/remove/main/preprocess.c @@ -5,7 +5,7 @@ extern "C" { #endif #ifndef _di_kt_remove_preprocess_file_ - uint16_t kt_remove_preprocess_file(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate) { + uint16_t kt_remove_preprocess_file(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate) { if (!main) return 0; @@ -21,7 +21,7 @@ extern "C" { main->setting.state.status = f_file_exists(path, main->setting.flag & kt_remove_main_flag_follow_d); - uint16_t flag_out = (main->setting.flag & kt_remove_main_flag_option_used_d) ? 0 : kt_remove_flag_file_operate_remove_d; + uint32_t flag_out = (main->setting.flag & kt_remove_main_flag_option_used_d) ? 0 : kt_remove_flag_file_operate_remove_d; if (main->setting.state.status == F_true) { const f_status_t status = f_file_is(path, F_file_type_link_d, F_false); @@ -402,7 +402,7 @@ extern "C" { #endif // _di_kt_remove_preprocess_file_ #ifndef _di_kt_remove_preprocess_file_recurse_ - f_status_t kt_remove_preprocess_file_recurse(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate) { + f_status_t kt_remove_preprocess_file_recurse(kt_remove_main_t * const main, const f_string_static_t path, const uint32_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; @@ -451,7 +451,7 @@ extern "C" { #endif // _di_kt_remove_preprocess_file_recurse_action_ #ifndef _di_kt_remove_preprocess_file_dates_ - void kt_remove_preprocess_file_dates(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate, const struct stat statistics) { + void kt_remove_preprocess_file_dates(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate, const struct stat statistics) { if (!main) return; diff --git a/sources/c/program/kevux/tools/remove/main/preprocess.h b/sources/c/program/kevux/tools/remove/main/preprocess.h index c27b951..61c6ea4 100644 --- a/sources/c/program/kevux/tools/remove/main/preprocess.h +++ b/sources/c/program/kevux/tools/remove/main/preprocess.h @@ -44,7 +44,7 @@ extern "C" { * @see kt_remove_preprocess_file_dates() */ #ifndef _di_kt_remove_preprocess_file_ - extern uint16_t kt_remove_preprocess_file(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate); + extern uint16_t kt_remove_preprocess_file(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate); #endif // _di_kt_remove_preprocess_file_ /** @@ -76,7 +76,7 @@ extern "C" { * @see fl_directory_do() */ #ifndef _di_kt_remove_preprocess_file_recurse_ - extern f_status_t kt_remove_preprocess_file_recurse(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate); + extern f_status_t kt_remove_preprocess_file_recurse(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate); #endif // _di_kt_remove_preprocess_file_recurse_ /** @@ -120,7 +120,7 @@ extern "C" { * The already loaded file statistics. */ #ifndef _di_kt_remove_preprocess_file_dates_ - extern void kt_remove_preprocess_file_dates(kt_remove_main_t * const main, const f_string_static_t path, const uint16_t flag_operate, const struct stat statistics); + extern void kt_remove_preprocess_file_dates(kt_remove_main_t * const main, const f_string_static_t path, const uint32_t flag_operate, const struct stat statistics); #endif // _di_kt_remove_preprocess_file_dates_ #ifdef __cplusplus diff --git a/sources/c/program/kevux/tools/remove/main/print/debug.c b/sources/c/program/kevux/tools/remove/main/print/debug.c index 904e60d..a609d7d 100644 --- a/sources/c/program/kevux/tools/remove/main/print/debug.c +++ b/sources/c/program/kevux/tools/remove/main/print/debug.c @@ -5,7 +5,7 @@ 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) { + f_status_t kt_remove_print_debug_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint32_t flag) { if (!print || !print->custom) return F_status_set_error(F_output_not); diff --git a/sources/c/program/kevux/tools/remove/main/print/debug.h b/sources/c/program/kevux/tools/remove/main/print/debug.h index ff68a71..dfb6733 100644 --- a/sources/c/program/kevux/tools/remove/main/print/debug.h +++ b/sources/c/program/kevux/tools/remove/main/print/debug.h @@ -39,7 +39,7 @@ extern "C" { * 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); + extern f_status_t kt_remove_print_debug_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint32_t flag); #endif // _di_kt_remove_print_debug_operate_file_remove_ #ifdef __cplusplus diff --git a/sources/c/program/kevux/tools/remove/main/print/message.c b/sources/c/program/kevux/tools/remove/main/print/message.c index f28ba86..adc11de 100644 --- a/sources/c/program/kevux/tools/remove/main/print/message.c +++ b/sources/c/program/kevux/tools/remove/main/print/message.c @@ -182,6 +182,22 @@ extern "C" { } #endif // _di_kt_remove_print_message_help_ +#ifndef _di_kt_remove_print_message_remove_confirm_ + f_status_t kt_remove_print_message_remove_confirm(fl_print_t * const print, const f_string_static_t path) { + + if (!print || !print->custom) return F_status_set_error(F_output_not); + + f_file_stream_lock(print->to); + + fl_print_format("%rRemove '%[%Q%]' (y/n): ", print->to, f_string_eol_s, print->set->notable, path, print->set->notable); + + f_file_stream_flush(print->to); + f_file_stream_unlock(print->to); + + return F_okay; + } +#endif // _di_kt_remove_print_message_remove_confirm_ + #ifdef __cplusplus } // extern "C" #endif diff --git a/sources/c/program/kevux/tools/remove/main/print/message.h b/sources/c/program/kevux/tools/remove/main/print/message.h index e6b4360..308bca7 100644 --- a/sources/c/program/kevux/tools/remove/main/print/message.h +++ b/sources/c/program/kevux/tools/remove/main/print/message.h @@ -49,6 +49,30 @@ extern "C" { extern f_status_t kt_remove_print_message_help(fl_print_t * const print, const f_color_context_t context); #endif // _di_kt_remove_print_message_help_ +/** + * Print message asking whether to confirm or deny removal of a path. + * + * @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 remove. + * + * @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_message_remove_confirm_ + extern f_status_t kt_remove_print_message_remove_confirm(fl_print_t * const print, const f_string_static_t path); +#endif // _di_kt_remove_print_message_remove_confirm_ + #ifdef __cplusplus } // extern "C" #endif diff --git a/sources/c/program/kevux/tools/remove/main/print/simulate.c b/sources/c/program/kevux/tools/remove/main/print/simulate.c index 2af5de5..e56b07c 100644 --- a/sources/c/program/kevux/tools/remove/main/print/simulate.c +++ b/sources/c/program/kevux/tools/remove/main/print/simulate.c @@ -72,7 +72,7 @@ extern "C" { #endif // _di_kt_remove_print_simulate_operate_date_ #ifndef _di_kt_remove_print_simulate_operate_file_ - f_status_t 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 uint32_t flag) { if (!print || !print->custom) return F_status_set_error(F_output_not); @@ -97,7 +97,7 @@ extern "C" { #endif // _di_kt_remove_print_simulate_operate_file_ #ifndef _di_kt_remove_print_simulate_operate_file_exists_ - 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) { + f_status_t kt_remove_print_simulate_operate_file_exists(fl_print_t * const print, const f_string_static_t path, const uint32_t flag) { if (!print || !print->custom) return F_status_set_error(F_output_not); diff --git a/sources/c/program/kevux/tools/remove/main/print/simulate.h b/sources/c/program/kevux/tools/remove/main/print/simulate.h index a1ee963..9a9ecf2 100644 --- a/sources/c/program/kevux/tools/remove/main/print/simulate.h +++ b/sources/c/program/kevux/tools/remove/main/print/simulate.h @@ -130,7 +130,7 @@ extern "C" { * F_output_not (with error bit) if setting is NULL. */ #ifndef _di_kt_remove_print_simulate_operate_file_ - 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); + extern f_status_t kt_remove_print_simulate_operate_file(fl_print_t * const print, const f_string_static_t path, const uint32_t flag); #endif // _di_kt_remove_print_simulate_operate_file_ /** @@ -159,7 +159,7 @@ extern "C" { * F_output_not (with error bit) if setting is NULL. */ #ifndef _di_kt_remove_print_simulate_operate_file_exists_ - 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); + extern f_status_t kt_remove_print_simulate_operate_file_exists(fl_print_t * const print, const f_string_static_t path, const uint32_t flag); #endif // _di_kt_remove_print_simulate_operate_file_exists_ /** diff --git a/sources/c/program/kevux/tools/remove/main/print/verbose.c b/sources/c/program/kevux/tools/remove/main/print/verbose.c index 0c605a1..5eb6029 100644 --- a/sources/c/program/kevux/tools/remove/main/print/verbose.c +++ b/sources/c/program/kevux/tools/remove/main/print/verbose.c @@ -5,7 +5,7 @@ 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) { + f_status_t kt_remove_print_verbose_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint32_t flag) { if (!print || !print->custom) return F_status_set_error(F_output_not); diff --git a/sources/c/program/kevux/tools/remove/main/print/verbose.h b/sources/c/program/kevux/tools/remove/main/print/verbose.h index d8051c5..0dfbfee 100644 --- a/sources/c/program/kevux/tools/remove/main/print/verbose.h +++ b/sources/c/program/kevux/tools/remove/main/print/verbose.h @@ -39,7 +39,7 @@ extern "C" { * 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); + extern f_status_t kt_remove_print_verbose_operate_file_remove(fl_print_t * const print, const f_string_static_t path, const uint32_t flag); #endif // _di_kt_remove_print_verbose_operate_file_remove_ #ifdef __cplusplus diff --git a/sources/c/program/kevux/tools/remove/main/remove.c b/sources/c/program/kevux/tools/remove/main/remove.c index 58e376f..0e3b499 100644 --- a/sources/c/program/kevux/tools/remove/main/remove.c +++ b/sources/c/program/kevux/tools/remove/main/remove.c @@ -66,7 +66,15 @@ extern "C" { f_print_dynamic(f_string_eol_s, main->program.output.to); } - if (F_status_is_error(main->setting.state.status)) return; + if (F_status_is_error(main->setting.state.status)) { + + // If told to skip all, then do not return as an error. + if (F_status_set_fine(main->setting.state.status) == F_skip) { + main->setting.state.status = F_okay; + } + + return; + } } // for if (main->cache.tree.used) { diff --git a/tests/unit/remove/c/test-remove-date_accessed.c b/tests/unit/remove/c/test-remove-date_accessed.c new file mode 100644 index 0000000..3a110db --- /dev/null +++ b/tests/unit/remove/c/test-remove-date_accessed.c @@ -0,0 +1,780 @@ +#include "test-remove.h" +#include "test-remove-date_accessed.h" + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +void test__kt_remove__date_accessed__date_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 18; + const uint8_t params_total = 15; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameters[] = { + kt_remove_date_format_example_00_s, + kt_remove_date_format_example_01_s, + kt_remove_date_format_example_02_s, + kt_remove_date_format_example_03_s, + kt_remove_date_format_example_04_s, + kt_remove_date_format_example_05_s, + kt_remove_date_format_example_06_s, + kt_remove_date_format_example_07_s, + kt_remove_date_format_example_08_s, + kt_remove_date_format_example_09_s, + kt_remove_date_format_example_10_s, + kt_remove_date_format_example_11_s, + kt_remove_date_format_example_12_s, + kt_remove_date_format_example_13_s, + kt_remove_date_format_example_14_s, + }; + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + }; + + bool time_spec_removes[] = { + F_false, // Equal + F_false, + F_false, + F_true, // Less + F_false, + F_false, + F_true, // Less Equal + F_false, + F_false, + F_false, // More + F_true, + F_true, + F_false, // More Equal + F_true, + F_true, + F_true, // Not + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + uint8_t param = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + for (param = 0; param < params_total; ++param) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_accessed_s, operators[date].string, parameters[param].string, 0 }; + + stats[type].st_atim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } // for + } +} + +void test__kt_remove__date_accessed__now_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("now", 0, 3); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 123); + + bool time_spec_removes[] = { + F_true, // Equal + F_false, + F_false, + F_false, + F_false, // Less + F_true, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_false, + F_true, + F_false, // More + F_false, + F_true, + F_false, + F_true, // More Equal + F_false, + F_true, + F_false, + F_false, // Not + F_true, + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_accessed_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_atim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +void test__kt_remove__date_accessed__today_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("today", 0, 5); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 123); + + bool time_spec_removes[] = { + F_true, // Equal + F_true, + F_false, + F_false, + F_false, // Less + F_false, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_false, + F_true, + F_false, // More + F_false, + F_true, + F_false, + F_true, // More Equal + F_true, + F_true, + F_false, + F_false, // Not + F_false, + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_accessed_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_atim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +void test__kt_remove__date_accessed__tomorrow_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("tomorrow", 0, 8); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 123); + + bool time_spec_removes[] = { + F_false, // Equal + F_false, + F_true, + F_false, + F_true, // Less + F_true, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_true, + F_true, + F_false, // More + F_false, + F_false, + F_false, + F_false, // More Equal + F_false, + F_true, + F_false, + F_true, // Not + F_true, + F_false, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_accessed_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_atim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +void test__kt_remove__date_accessed__yesterday_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("yesterday", 0, 9); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 345); + + bool time_spec_removes[] = { + F_false, // Equal + F_false, + F_false, + F_true, + F_false, // Less + F_false, + F_false, + F_false, + F_false, // Less Equal + F_false, + F_false, + F_true, + F_true, // More + F_true, + F_true, + F_false, + F_true, // More Equal + F_true, + F_true, + F_true, + F_true, // Not + F_true, + F_true, + F_false, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_accessed_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_atim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +#ifdef __cplusplus +} // extern "C" +#endif diff --git a/tests/unit/remove/c/test-remove-date_accessed.h b/tests/unit/remove/c/test-remove-date_accessed.h new file mode 100644 index 0000000..1627a06 --- /dev/null +++ b/tests/unit/remove/c/test-remove-date_accessed.h @@ -0,0 +1,38 @@ +/** + * Kevux Tools - Remove + * + * Project: Kevux Tools + * API Version: 0.5 + * Licenses: lgpl-2.1-or-later + * + * Test the remove. + */ +#ifndef _TEST__KT_remove__date_accessed +#define _TEST__KT_remove__date_accessed + +/** + * Test that the remove works when the --changed parameter with date parameters passed. + */ +extern void test__kt_remove__date_accessed__date_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "now" parameter passed. + */ +extern void test__kt_remove__date_accessed__now_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "today" parameter passed. + */ +extern void test__kt_remove__date_accessed__today_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "tomorrow" parameter passed. + */ +extern void test__kt_remove__date_accessed__tomorrow_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "yesterday" parameter passed. + */ +extern void test__kt_remove__date_accessed__yesterday_works(void **state); + +#endif // _TEST__KT_remove__date_accessed diff --git a/tests/unit/remove/c/test-remove-date_changed.c b/tests/unit/remove/c/test-remove-date_changed.c index 6f5ab60..3a8cb03 100644 --- a/tests/unit/remove/c/test-remove-date_changed.c +++ b/tests/unit/remove/c/test-remove-date_changed.c @@ -119,7 +119,7 @@ void test__kt_remove__date_changed__date_works(void **state) { for (; type < types_total; ++type) { - for (date = 12; date < dates_total; ++date) { + for (date = 0; date < dates_total; ++date) { for (param = 0; param < params_total; ++param) { diff --git a/tests/unit/remove/c/test-remove-date_updated.c b/tests/unit/remove/c/test-remove-date_updated.c new file mode 100644 index 0000000..648be21 --- /dev/null +++ b/tests/unit/remove/c/test-remove-date_updated.c @@ -0,0 +1,780 @@ +#include "test-remove.h" +#include "test-remove-date_updated.h" + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +void test__kt_remove__date_updated__date_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 18; + const uint8_t params_total = 15; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameters[] = { + kt_remove_date_format_example_00_s, + kt_remove_date_format_example_01_s, + kt_remove_date_format_example_02_s, + kt_remove_date_format_example_03_s, + kt_remove_date_format_example_04_s, + kt_remove_date_format_example_05_s, + kt_remove_date_format_example_06_s, + kt_remove_date_format_example_07_s, + kt_remove_date_format_example_08_s, + kt_remove_date_format_example_09_s, + kt_remove_date_format_example_10_s, + kt_remove_date_format_example_11_s, + kt_remove_date_format_example_12_s, + kt_remove_date_format_example_13_s, + kt_remove_date_format_example_14_s, + }; + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * kt_remove_time_seconds_in_year_d, 789), + macro_f_time_spec_t_initialize_1(1672429502, 0), + }; + + bool time_spec_removes[] = { + F_false, // Equal + F_false, + F_false, + F_true, // Less + F_false, + F_false, + F_true, // Less Equal + F_false, + F_false, + F_false, // More + F_true, + F_true, + F_false, // More Equal + F_true, + F_true, + F_true, // Not + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + uint8_t param = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + for (param = 0; param < params_total; ++param) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_updated_s, operators[date].string, parameters[param].string, 0 }; + + stats[type].st_mtim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } // for + } +} + +void test__kt_remove__date_updated__now_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("now", 0, 3); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 123); + + bool time_spec_removes[] = { + F_true, // Equal + F_false, + F_false, + F_false, + F_false, // Less + F_true, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_false, + F_true, + F_false, // More + F_false, + F_true, + F_false, + F_true, // More Equal + F_false, + F_true, + F_false, + F_false, // Not + F_true, + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_updated_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_mtim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +void test__kt_remove__date_updated__today_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("today", 0, 5); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 123); + + bool time_spec_removes[] = { + F_true, // Equal + F_true, + F_false, + F_false, + F_false, // Less + F_false, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_false, + F_true, + F_false, // More + F_false, + F_true, + F_false, + F_true, // More Equal + F_true, + F_true, + F_false, + F_false, // Not + F_false, + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_updated_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_mtim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +void test__kt_remove__date_updated__tomorrow_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("tomorrow", 0, 8); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 123); + + bool time_spec_removes[] = { + F_false, // Equal + F_false, + F_true, + F_false, + F_true, // Less + F_true, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_true, + F_true, + F_false, // More + F_false, + F_false, + F_false, + F_false, // More Equal + F_false, + F_true, + F_false, + F_true, // Not + F_true, + F_false, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_updated_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_mtim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +void test__kt_remove__date_updated__yesterday_works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("yesterday", 0, 9); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + const f_time_spec_t time_spec_clocks = macro_f_time_spec_t_initialize_1(2 * 86400, 345); + + bool time_spec_removes[] = { + F_false, // Equal + F_false, + F_false, + F_true, + F_false, // Less + F_false, + F_false, + F_false, + F_false, // Less Equal + F_false, + F_false, + F_true, + F_true, // More + F_true, + F_true, + F_false, + F_true, // More Equal + F_true, + F_true, + F_true, + F_true, // Not + F_true, + F_true, + F_false, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_updated_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_mtim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + will_return(__wrap_f_time_clock_get, &time_spec_clocks); + will_return(__wrap_f_time_clock_get, F_okay); + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +#ifdef __cplusplus +} // extern "C" +#endif diff --git a/tests/unit/remove/c/test-remove-date_updated.h b/tests/unit/remove/c/test-remove-date_updated.h new file mode 100644 index 0000000..96b93ad --- /dev/null +++ b/tests/unit/remove/c/test-remove-date_updated.h @@ -0,0 +1,38 @@ +/** + * Kevux Tools - Remove + * + * Project: Kevux Tools + * API Version: 0.5 + * Licenses: lgpl-2.1-or-later + * + * Test the remove. + */ +#ifndef _TEST__KT_remove__date_updated +#define _TEST__KT_remove__date_updated + +/** + * Test that the remove works when the --changed parameter with date parameters passed. + */ +extern void test__kt_remove__date_updated__date_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "now" parameter passed. + */ +extern void test__kt_remove__date_updated__now_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "today" parameter passed. + */ +extern void test__kt_remove__date_updated__today_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "tomorrow" parameter passed. + */ +extern void test__kt_remove__date_updated__tomorrow_works(void **state); + +/** + * Test that the remove works when the --changed parameter with "yesterday" parameter passed. + */ +extern void test__kt_remove__date_updated__yesterday_works(void **state); + +#endif // _TEST__KT_remove__date_updated diff --git a/tests/unit/remove/c/test-remove-unix.c b/tests/unit/remove/c/test-remove-unix.c new file mode 100644 index 0000000..695e9f2 --- /dev/null +++ b/tests/unit/remove/c/test-remove-unix.c @@ -0,0 +1,161 @@ +#include "test-remove.h" +#include "test-remove-unix.h" + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +void test__kt_remove__unix__works(void **state) { + + mock_unwrap = 0; + mock_unwrap_f_time_clock_get = 0; + + const f_string_static_t target = macro_f_string_static_t_initialize_1("to_remove", 0, 9); + + const uint8_t types_total = 8; + const uint8_t dates_total = 24; + + struct stat stats[types_total]; + + memset(stats, 0, sizeof(struct stat) * types_total); + + stats[0].st_mode = F_file_mode_all_d | F_file_type_block_d; + stats[1].st_mode = F_file_mode_all_d | F_file_type_character_d; + stats[2].st_mode = F_file_mode_all_d | F_file_type_directory_d; + stats[3].st_mode = F_file_mode_all_d | F_file_type_fifo_d; + stats[4].st_mode = F_file_mode_all_d | F_file_type_link_d; + stats[5].st_mode = F_file_mode_all_d | F_file_type_regular_d; + stats[6].st_mode = F_file_mode_all_d | F_file_type_socket_d; + stats[7].st_mode = F_file_mode_all_d & ~S_IFMT; + + const f_string_static_t parameter = macro_f_string_static_t_initialize_1("172800", 0, 6); + + const f_string_static_t operators[] = { + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_equal_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_less_equal_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_more_equal_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + kt_remove_date_symbol_not_s, + }; + + const f_time_spec_t time_spec_dates[] = { + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Less Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // More Equal + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + macro_f_time_spec_t_initialize_1(2 * 86400, 345), // Not + macro_f_time_spec_t_initialize_1(2 * 86400, 0), + macro_f_time_spec_t_initialize_1(3 * 86400, 10), + macro_f_time_spec_t_initialize_1(1 * 86400, 4000), + }; + + bool time_spec_removes[] = { + F_true, // Equal + F_true, + F_false, + F_false, + F_false, // Less + F_false, + F_false, + F_true, + F_true, // Less Equal + F_true, + F_false, + F_true, + F_false, // More + F_false, + F_true, + F_false, + F_true, // More Equal + F_true, + F_true, + F_false, + F_false, // Not + F_false, + F_true, + F_true, + }; + + { + uint8_t type = 0; + uint8_t date = 0; + + for (; type < types_total; ++type) { + + for (date = 0; date < dates_total; ++date) { + + const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_changed_s, operators[date].string, parameter.string, 0 }; + + stats[type].st_ctim = time_spec_dates[date]; + + // Pre-process file. + will_return(__wrap_f_file_exists, F_true); + will_return(__wrap_f_file_is, macro_f_file_type_is_link(stats[type].st_mode)); + will_return(__wrap_f_file_stat, &stats[type]); + will_return(__wrap_f_file_stat, F_okay); + + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_f_directory_empty, F_true); + } + + if (time_spec_removes[date]) { + if (macro_f_file_type_is_directory(stats[type].st_mode)) { + will_return(__wrap_fl_directory_do, 1); + 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); + } + else { + will_return(__wrap_f_file_remove, F_okay); + } + } + + const int result = kt_main_test__remove(5, argv, 0); + + assert_int_equal(result, 0); + } // for + } // for + } +} + +#ifdef __cplusplus +} // extern "C" +#endif diff --git a/tests/unit/remove/c/test-remove-unix.h b/tests/unit/remove/c/test-remove-unix.h new file mode 100644 index 0000000..6e3d8d5 --- /dev/null +++ b/tests/unit/remove/c/test-remove-unix.h @@ -0,0 +1,18 @@ +/** + * Kevux Tools - Remove + * + * Project: Kevux Tools + * API Version: 0.5 + * Licenses: lgpl-2.1-or-later + * + * Test the remove. + */ +#ifndef _TEST__KT_remove__unix +#define _TEST__KT_remove__unix + +/** + * Test that the remove works when the --changed parameter with date parameters passed. + */ +extern void test__kt_remove__unix__works(void **state); + +#endif // _TEST__KT_remove__unix diff --git a/tests/unit/remove/c/test-remove.c b/tests/unit/remove/c/test-remove.c index c5f3a5c..b368c15 100644 --- a/tests/unit/remove/c/test-remove.c +++ b/tests/unit/remove/c/test-remove.c @@ -22,12 +22,24 @@ int main(void) { cmocka_unit_test(test__kt_remove__print_help__works), cmocka_unit_test(test__kt_remove__print_version__works), + cmocka_unit_test(test__kt_remove__date_accessed__date_works), + cmocka_unit_test(test__kt_remove__date_accessed__now_works), + cmocka_unit_test(test__kt_remove__date_accessed__today_works), + cmocka_unit_test(test__kt_remove__date_accessed__tomorrow_works), + cmocka_unit_test(test__kt_remove__date_accessed__yesterday_works), + cmocka_unit_test(test__kt_remove__date_changed__date_works), cmocka_unit_test(test__kt_remove__date_changed__now_works), cmocka_unit_test(test__kt_remove__date_changed__today_works), cmocka_unit_test(test__kt_remove__date_changed__tomorrow_works), cmocka_unit_test(test__kt_remove__date_changed__yesterday_works), + cmocka_unit_test(test__kt_remove__date_updated__date_works), + cmocka_unit_test(test__kt_remove__date_updated__now_works), + cmocka_unit_test(test__kt_remove__date_updated__today_works), + cmocka_unit_test(test__kt_remove__date_updated__tomorrow_works), + cmocka_unit_test(test__kt_remove__date_updated__yesterday_works), + cmocka_unit_test(test__kt_remove__directory_no_args__one_empty_exists_link), cmocka_unit_test(test__kt_remove__directory_no_args__one_empty_exists_link_not), cmocka_unit_test(test__kt_remove__directory_no_args__one_empty_exists_not), @@ -69,6 +81,8 @@ int main(void) { cmocka_unit_test(test__kt_remove__regular_no_args__two_exists_link_not), cmocka_unit_test(test__kt_remove__regular_no_args__two_exists_not), + cmocka_unit_test(test__kt_remove__unix__works), + cmocka_unit_test(test__kt_remove__user__different_works), cmocka_unit_test(test__kt_remove__user__name_works), cmocka_unit_test(test__kt_remove__user__same_works), diff --git a/tests/unit/remove/c/test-remove.h b/tests/unit/remove/c/test-remove.h index f882a81..7cf84cf 100644 --- a/tests/unit/remove/c/test-remove.h +++ b/tests/unit/remove/c/test-remove.h @@ -30,7 +30,9 @@ #include "main-test-remove.h" // Test includes. +#include "test-remove-date_accessed.h" #include "test-remove-date_changed.h" +#include "test-remove-date_updated.h" #include "test-remove-directory_no_args.h" #include "test-remove-directory_recurse_simple.h" #include "test-remove-directory_tree_simple.h" @@ -41,6 +43,7 @@ #include "test-remove-print_help.h" #include "test-remove-print_version.h" #include "test-remove-regular_no_args.h" +#include "test-remove-unix.h" #include "test-remove-user.h" #ifdef __cplusplus -- 1.8.3.1