if (main->setting.flag & kt_remove_main_flag_mode_d) {
const mode_t mode = statistics.st_mode & F_file_mode_all_d;
+ const uint8_t types[] = {
+ kt_remove_flag_mode_different_d,
+ kt_remove_flag_mode_same_d,
+ kt_remove_flag_mode_similar_d,
+ kt_remove_flag_mode_not_d,
+ };
+
+ const f_string_static_t strings[] = {
+ kt_remove_mode_word_different_s,
+ kt_remove_mode_word_same_s,
+ kt_remove_mode_word_similar_s,
+ kt_remove_mode_word_not_s,
+ };
+
fll_print_format(" mode %@03un%r", print->to, (f_number_unsigned_t) mode, f_string_eol_s);
for (i = 0; i < main->setting.modes.used; ++i) {
if (main->setting.modes.array[i].type == kt_remove_flag_mode_different_d) {
- if (main->setting.modes.array[i].mode & ~mode) break;
+ if (main->setting.modes.array[i].mode ^ mode) break;
}
else if (main->setting.modes.array[i].type == kt_remove_flag_mode_same_d) {
- if (main->setting.modes.array[i].mode == mode) break;
+ if (!(main->setting.modes.array[i].mode ^ mode)) break;
}
else if (main->setting.modes.array[i].type == kt_remove_flag_mode_similar_d) {
if (main->setting.modes.array[i].mode & mode) break;
} // for
if (i < main->setting.modes.used) {
- uint8_t types[] = {
- kt_remove_flag_mode_different_d,
- kt_remove_flag_mode_same_d,
- kt_remove_flag_mode_similar_d,
- kt_remove_flag_mode_not_d,
- };
-
- f_string_static_t strings[] = {
- kt_remove_mode_word_different_s,
- kt_remove_mode_word_same_s,
- kt_remove_mode_word_similar_s,
- kt_remove_mode_word_not_s,
- };
-
for (j = 0; j < 4; ++j) {
if (main->setting.modes.array[i].type == types[j]) {
--- /dev/null
+#include "test-remove.h"
+#include "test-remove-file_mode.h"
+
+#include <program/kevux/tools/remove/remove/main.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+void test__kt_remove__file_mode__different_works(void **state) {
+
+ mock_unwrap = 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 modes_total = 10; // For both modes and params.
+
+ const int32_t types[] = {
+ F_file_type_block_d,
+ F_file_type_character_d,
+ F_file_type_directory_d,
+ F_file_type_fifo_d,
+ F_file_type_link_d,
+ F_file_type_regular_d,
+ F_file_type_socket_d,
+ ~S_IFMT,
+ };
+
+ const int32_t modes[] = {
+ F_file_mode_all_rwx_d,
+ F_file_mode_owner_r_d,
+ F_file_mode_owner_w_d,
+ F_file_mode_owner_x_d,
+ F_file_mode_group_r_d,
+ F_file_mode_group_w_d,
+ F_file_mode_group_x_d,
+ F_file_mode_world_r_d,
+ F_file_mode_world_w_d,
+ F_file_mode_world_x_d,
+ 0,
+ };
+
+ const f_string_t params[] = {
+ "ugo+rwx",
+ "u+r",
+ "u+w",
+ "u+x",
+ "g+r",
+ "g+w",
+ "g+x",
+ "o+r",
+ "o+w",
+ "o+x",
+ "0",
+ };
+
+ {
+ uint8_t type = 0;
+ uint8_t mode = 0;
+ uint8_t param = 0;
+
+ for (type = 0; type < types_total; ++type) {
+
+ for (mode = 0; mode < modes_total; ++mode) {
+
+ for (param = 0; param < modes_total; ++param) {
+
+ const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_mode_s, kt_remove_mode_word_different_s.string, params[param], 0 };
+
+ struct stat stat_file;
+
+ memset(&stat_file, 0, sizeof(struct stat));
+
+ // The unknown type at the last index requires '&' logic rather than '|' logic.
+ stat_file.st_mode = types[type] == ~S_IFMT
+ ? modes[mode] & types[type]
+ : modes[mode] | types[type];
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, macro_f_file_type_is_link(stat_file.st_mode));
+ will_return(__wrap_f_file_stat, &stat_file);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Process file.
+ if (param != mode) {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ 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);
+ }
+ }
+ else {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ }
+ }
+
+ const int result = kt_main_test__remove(5, argv, 0);
+
+ assert_int_equal(result, 0);
+ } // for
+ } // for
+ } // for
+ }
+}
+
+void test__kt_remove__file_mode__same_works(void **state) {
+
+ mock_unwrap = 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 modes_total = 10; // For both modes and params.
+
+ const int32_t types[] = {
+ F_file_type_block_d,
+ F_file_type_character_d,
+ F_file_type_directory_d,
+ F_file_type_fifo_d,
+ F_file_type_link_d,
+ F_file_type_regular_d,
+ F_file_type_socket_d,
+ ~S_IFMT,
+ };
+
+ const int32_t modes[] = {
+ F_file_mode_all_rwx_d,
+ F_file_mode_owner_r_d,
+ F_file_mode_owner_w_d,
+ F_file_mode_owner_x_d,
+ F_file_mode_group_r_d,
+ F_file_mode_group_w_d,
+ F_file_mode_group_x_d,
+ F_file_mode_world_r_d,
+ F_file_mode_world_w_d,
+ F_file_mode_world_x_d,
+ 0,
+ };
+
+ const f_string_t params[] = {
+ "ugo+rwx",
+ "u+r",
+ "u+w",
+ "u+x",
+ "g+r",
+ "g+w",
+ "g+x",
+ "o+r",
+ "o+w",
+ "o+x",
+ "0",
+ };
+
+ {
+ uint8_t type = 0;
+ uint8_t mode = 0;
+ uint8_t param = 0;
+
+ for (type = 0; type < types_total; ++type) {
+
+ for (mode = 0; mode < modes_total; ++mode) {
+
+ for (param = 0; param < modes_total; ++param) {
+
+ const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_mode_s, kt_remove_mode_word_same_s.string, params[param], 0 };
+
+ struct stat stat_file;
+
+ memset(&stat_file, 0, sizeof(struct stat));
+
+ // The unknown type at the last index requires '&' logic rather than '|' logic.
+ stat_file.st_mode = types[type] == ~S_IFMT
+ ? modes[mode] & types[type]
+ : modes[mode] | types[type];
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, macro_f_file_type_is_link(stat_file.st_mode));
+ will_return(__wrap_f_file_stat, &stat_file);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Process file.
+ if (param == mode) {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ 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);
+ }
+ }
+ else {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ }
+ }
+
+ const int result = kt_main_test__remove(5, argv, 0);
+
+ assert_int_equal(result, 0);
+ } // for
+ } // for
+ } // for
+ }
+}
+
+void test__kt_remove__file_mode__similar_works(void **state) {
+
+ mock_unwrap = 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 modes_total = 10; // For both modes and params.
+
+ const int32_t types[] = {
+ F_file_type_block_d,
+ F_file_type_character_d,
+ F_file_type_directory_d,
+ F_file_type_fifo_d,
+ F_file_type_link_d,
+ F_file_type_regular_d,
+ F_file_type_socket_d,
+ ~S_IFMT,
+ };
+
+ const int32_t modes[] = {
+ F_file_mode_all_rwx_d,
+ F_file_mode_owner_r_d,
+ F_file_mode_owner_w_d,
+ F_file_mode_owner_x_d,
+ F_file_mode_group_r_d,
+ F_file_mode_group_w_d,
+ F_file_mode_group_x_d,
+ F_file_mode_world_r_d,
+ F_file_mode_world_w_d,
+ F_file_mode_world_x_d,
+ 0,
+ };
+
+ const f_string_t params[] = {
+ "ugo+rwx",
+ "u+r",
+ "u+w",
+ "u+x",
+ "g+r",
+ "g+w",
+ "g+x",
+ "o+r",
+ "o+w",
+ "o+x",
+ "0",
+ };
+
+ {
+ uint8_t type = 0;
+ uint8_t mode = 0;
+ uint8_t param = 0;
+
+ for (type = 0; type < types_total; ++type) {
+
+ for (mode = 0; mode < modes_total; ++mode) {
+
+ for (param = 0; param < modes_total; ++param) {
+
+ const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_mode_s, kt_remove_mode_word_similar_s.string, params[param], 0 };
+
+ struct stat stat_file;
+
+ memset(&stat_file, 0, sizeof(struct stat));
+
+ // The unknown type at the last index requires '&' logic rather than '|' logic.
+ stat_file.st_mode = types[type] == ~S_IFMT
+ ? modes[mode] & types[type]
+ : modes[mode] | types[type];
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, macro_f_file_type_is_link(stat_file.st_mode));
+ will_return(__wrap_f_file_stat, &stat_file);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Process file.
+ if (modes[param] & modes[mode]) {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ 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);
+ }
+ }
+ else {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ }
+ }
+
+ const int result = kt_main_test__remove(5, argv, 0);
+
+ assert_int_equal(result, 0);
+ } // for
+ } // for
+ } // for
+ }
+}
+
+void test__kt_remove__file_mode__not_works(void **state) {
+
+ mock_unwrap = 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 modes_total = 10; // For both modes and params.
+
+ const int32_t types[] = {
+ F_file_type_block_d,
+ F_file_type_character_d,
+ F_file_type_directory_d,
+ F_file_type_fifo_d,
+ F_file_type_link_d,
+ F_file_type_regular_d,
+ F_file_type_socket_d,
+ ~S_IFMT,
+ };
+
+ const int32_t modes[] = {
+ F_file_mode_all_rwx_d,
+ F_file_mode_owner_r_d,
+ F_file_mode_owner_w_d,
+ F_file_mode_owner_x_d,
+ F_file_mode_group_r_d,
+ F_file_mode_group_w_d,
+ F_file_mode_group_x_d,
+ F_file_mode_world_r_d,
+ F_file_mode_world_w_d,
+ F_file_mode_world_x_d,
+ 0,
+ };
+
+ const f_string_t params[] = {
+ "ugo+rwx",
+ "u+r",
+ "u+w",
+ "u+x",
+ "g+r",
+ "g+w",
+ "g+x",
+ "o+r",
+ "o+w",
+ "o+x",
+ "0",
+ };
+
+ {
+ uint8_t type = 0;
+ uint8_t mode = 0;
+ uint8_t param = 0;
+
+ for (type = 0; type < types_total; ++type) {
+
+ for (mode = 0; mode < modes_total; ++mode) {
+
+ for (param = 0; param < modes_total; ++param) {
+
+ const f_string_t argv[] = { "mocked_main", target.string, "-" KT_REMOVE_short_mode_s, kt_remove_mode_word_not_s.string, params[param], 0 };
+
+ struct stat stat_file;
+
+ memset(&stat_file, 0, sizeof(struct stat));
+
+ // The unknown type at the last index requires '&' logic rather than '|' logic.
+ stat_file.st_mode = types[type] == ~S_IFMT
+ ? modes[mode] & types[type]
+ : modes[mode] | types[type];
+
+ // Pre-process file.
+ will_return(__wrap_f_file_exists, F_true);
+ will_return(__wrap_f_file_is, macro_f_file_type_is_link(stat_file.st_mode));
+ will_return(__wrap_f_file_stat, &stat_file);
+ will_return(__wrap_f_file_stat, F_okay);
+
+ // Process file.
+ if (modes[param] != modes[mode]) {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ 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);
+ }
+ }
+ else {
+ if (macro_f_file_type_is_directory(stat_file.st_mode)) {
+ will_return(__wrap_f_directory_empty, F_true);
+ }
+ }
+
+ const int result = kt_main_test__remove(5, argv, 0);
+
+ assert_int_equal(result, 0);
+ } // for
+ } // for
+ } // for
+ }
+}
+
+#ifdef __cplusplus
+} // extern "C"
+#endif