* - empty_not: Remove not empty directories.
* - empty_not_fail: Fail on not empty directories.
* - fifo: Remove by file type: FIFO.
- * - follow: Follow symbolic links deleting the file being pointed to rather than the link itself (when not set the link itself is deleted).
+ * - follow: Follow symbolic links for deleting the file being pointed to rather than the link itself (when not set the link itself is deleted).
* - force: Forcibly delete.
* - group: Remove by GID.
* - help: Print help.
* kt_remove_flag_file_operate_*_e:
* - none: No flags set.
* - directory: Is a directory.
+ * - follow: Follow the symbolic link.
* - recurse: Perform recursively (only on directories).
* - remove: Perform remove.
*/
enum {
kt_remove_flag_file_operate_none_e = 0x0,
kt_remove_flag_file_operate_directory_e = 0x1,
- kt_remove_flag_file_operate_recurse_e = 0x2,
- kt_remove_flag_file_operate_remove_e = 0x4,
+ kt_remove_flag_file_operate_follow_e = 0x2,
+ kt_remove_flag_file_operate_recurse_e = 0x4,
+ kt_remove_flag_file_operate_remove_e = 0x8,
}; // enum
#endif // _di_kt_remove_flag_file_operate_e_
f_number_unsigned_t i = 0;
uint8_t flag = (main->setting.flag & kt_remove_main_flag_option_used_e) ? 0 : kt_remove_flag_file_operate_remove_e;
+ if (main->setting.flag & kt_remove_main_flag_follow_e) {
+ flag |= kt_remove_flag_file_operate_follow_e;
+ }
+
struct stat statistics;
memset(&statistics, 0, sizeof(struct stat));
}
if (main->setting.flag & kt_remove_main_flag_block_e) {
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_block_d) {
+ if (macro_f_file_type_is_block(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_remove_e;
}
}
if (main->setting.flag & kt_remove_main_flag_character_e) {
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_character_d) {
+ if (macro_f_file_type_is_character(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_remove_e;
}
}
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_directory_d) {
+
+ if (macro_f_file_type_is_directory(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_directory_e;
if (main->setting.flag & kt_remove_main_flag_directory_e) {
}
if (main->setting.flag & kt_remove_main_flag_fifo_e) {
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_fifo_d) {
+ if (macro_f_file_type_is_fifo(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_remove_e;
}
}
if (main->setting.flag & kt_remove_main_flag_link_e) {
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_link_d) {
+ if (macro_f_file_type_is_link(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_remove_e;
}
}
if (main->setting.flag & kt_remove_main_flag_regular_e) {
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_regular_d) {
+ if (macro_f_file_type_is_regular(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_remove_e;
}
}
if (main->setting.flag & kt_remove_main_flag_socket_e) {
- if (macro_f_file_type_get(statistics.st_mode) == F_file_type_socket_d) {
+ if (macro_f_file_type_is_socket(statistics.st_mode)) {
flag |= kt_remove_flag_file_operate_remove_e;
}
}
kt_remove_print_simulate_operate_boolean(&main->program.output, kt_remove_recurse_s, flag & kt_remove_flag_file_operate_recurse_e);
}
- // @todo add check here to see if file is a directory, apply any appropriate restrictions (such as not removing non-empty directories without force or recursive).
- // @todo be sure too handle all of the remaining flags, such as tree, force, recurse, etc...:
- // - kt_remove_main_flag_empty_only_e
- // - kt_remove_main_flag_empty_only_fail_e
- // - kt_remove_main_flag_empty_not_e
- // - kt_remove_main_flag_empty_not_fail_e
- // - kt_remove_main_flag_prompt_all_e
- // - kt_remove_main_flag_prompt_follow_e
- // - kt_remove_main_flag_prompt_never_e
- // - kt_remove_main_flag_prompt_once_e
- // - kt_remove_main_flag_recurse_e
- // - kt_remove_main_flag_tree_e
+ if (flag & kt_remove_flag_file_operate_directory_e) {
+ if (main->setting.flag & kt_remove_main_flag_empty_all_d) {
+ // @todo handle simulate for this.
+ }
+
+ // Recurse effectively forces tree.
+ if (main->setting.flag & kt_remove_main_flag_recurse_e) {
+ // @todo handle simulate for this.
+ // @todo consider not following "rm" and having recurse not act like "--force" is specified.
+ }
+ else if (main->setting.flag & kt_remove_main_flag_tree_e) {
+ // @todo handle simulate for this.
+ }
+ }
+
+ if (main->setting.flag & kt_remove_main_flag_prompt_all_d) {
+ // @todo handle simulate for this.
+ }
kt_remove_print_simulate_operate_boolean(&main->program.output, kt_remove_remove_s, flag & kt_remove_flag_file_operate_remove_e);
- // @todo do actual removal.
+ if (!(main->setting.flag & kt_remove_main_flag_simulate_e)) {
+ if (flag & kt_remove_flag_file_operate_directory_e) {
+ kt_remove_operate_file_directory(main, path, flag);
+ if (F_status_is_error(main->setting.state.status)) return;
+ }
+ else {
+ kt_remove_operate_file_normal(main, path, flag);
+ if (F_status_is_error(main->setting.state.status)) return;
+ }
+ }
+
+ main->setting.state.status = F_none;
}
#endif // _di_kt_remove_operate_file_
+#ifndef _di_kt_remove_operate_file_directory_
+ void kt_remove_operate_file_directory(kt_remove_main_t * const main, const f_string_static_t path, const uint8_t flag) {
+
+ if (!(flag & kt_remove_flag_file_operate_remove_e)) {
+ main->setting.state.status = F_no;
+
+ return;
+ }
+
+ main->setting.state.status = F_yes;
+ }
+#endif // _di_kt_remove_operate_file_directory_
+
+#ifndef _di_kt_remove_operate_file_normal_
+ void kt_remove_operate_file_normal(kt_remove_main_t * const main, const f_string_static_t path, const uint8_t flag) {
+
+ if (!(flag & kt_remove_flag_file_operate_remove_e)) {
+ main->setting.state.status = F_no;
+
+ return;
+ }
+
+ // @todo consider providing a "follow deep" parameter for recursively following until a non-link is reached.
+ if (flag & kt_remove_flag_file_operate_follow_e) {
+ main->setting.buffer.used = 0;
+
+ main->setting.state.status = f_file_link_read(path, F_false, &main->setting.buffer);
+
+ if (F_status_is_error(main->setting.state.status)) {
+ kt_remove_print_error_file(&main->program.error, macro_kt_remove_f(f_file_remove), path, f_file_operation_stat_s, fll_error_file_type_link_e);
+
+ return;
+ }
+
+ main->setting.state.status = f_file_remove(main->setting.buffer);
+
+ if (F_status_is_error(main->setting.state.status)) {
+ kt_remove_print_error_file(&main->program.error, macro_kt_remove_f(f_file_remove), main->setting.buffer, f_file_operation_delete_s, fll_error_file_type_file_e);
+
+ return;
+ }
+ }
+ else {
+ main->setting.state.status = f_file_remove(path);
+
+ if (F_status_is_error(main->setting.state.status)) {
+ kt_remove_print_error_file(&main->program.error, macro_kt_remove_f(f_file_remove), path, f_file_operation_delete_s, fll_error_file_type_file_e);
+
+ return;
+ }
+ }
+
+ main->setting.state.status = F_yes;
+ }
+#endif // _di_kt_remove_operate_file_normal_
+
#ifdef __cplusplus
} // extern "C"
#endif
* This alters main.setting.state.status:
* F_none on success.
* F_data_not on success but file is an empty string.
- *
- * F_parameter (with error bit) if main is NULL or setting is NULL.
* @param path
* The path to the file to operate on.
*/
extern void kt_remove_operate_file(kt_remove_main_t * const main, const f_string_static_t path);
#endif // _di_kt_remove_operate_file_
+/**
+ * Perform actual file removal for directory files.
+ *
+ * @param main
+ * The main program and settings data.
+ *
+ * This alters main.setting.state.status:
+ * F_no on success but file is not to be removed.
+ * F_yes on success and file is removed.
+ *
+ * Errors (with error bit) from: f_file_remove().
+ * @param path
+ * The path to the file to operate on.
+ * @param flag
+ * The operate file specific flags from kt_remove_flag_file_operate_*_e.
+ */
+#ifndef _di_kt_remove_operate_file_directory_
+ extern void kt_remove_operate_file_directory(kt_remove_main_t * const main, const f_string_static_t path, const uint8_t flag);
+#endif // _di_kt_remove_operate_file_directory_
+
+/**
+ * Perform actual file removal for non-directory files.
+ *
+ * @param main
+ * The main program and settings data.
+ *
+ * This alters main.setting.state.status:
+ * F_no on success but file is not to be removed.
+ * F_yes on success and file is removed.
+ *
+ * Errors (with error bit) from: f_file_remove().
+ * @param path
+ * The path to the file to operate on.
+ * @param flag
+ * The operate file specific flags from kt_remove_flag_file_operate_*_e.
+ *
+ * @see f_file_remove()
+ */
+#ifndef _di_kt_remove_operate_file_normal_
+ extern void kt_remove_operate_file_normal(kt_remove_main_t * const main, const f_string_static_t path, const uint8_t flag);
+#endif // _di_kt_remove_operate_file_normal_
+
#ifdef __cplusplus
} // extern "C"
#endif