summary | shortlog | log | commit | commitdiff | tree
raw | patch | inline | side by side (parent: 4bac3d2)
raw | patch | inline | side by side (parent: 4bac3d2)
| author | Sebastian Harl <sh@tokkee.org> | |
| Mon, 30 Jun 2014 06:40:08 +0000 (08:40 +0200) | ||
| committer | Sebastian Harl <sh@tokkee.org> | |
| Mon, 30 Jun 2014 06:40:08 +0000 (08:40 +0200) | 
| src/include/utils/os.h | patch | blob | history | |
| src/utils/os.c | patch | blob | history | 
diff --git a/src/include/utils/os.h b/src/include/utils/os.h
index 26f7e59e6016137495fb0a5b2a94a630ddf7738e..2e6c54776ede89ba35d2f70aacda889df89b7d75 100644 (file)
--- a/src/include/utils/os.h
+++ b/src/include/utils/os.h
 /*
  * sysdb_mkdir_all:
  * Recursively create the directory 'pathname' (similar to 'mkdir -p' on the
- * command line) and ensure that its mode is 'mode'. Any newly created
- * directories will have permissions according the the specified mode
- * (modified by the process's umask in the usual way). The function will
- * always ensure the specified permissions for the specified directory, even
- * if it exists already.
+ * command line) using file permissions as specified by 'mode'.
  *
  * Returns:
  *  - 0 on success
 int
 sdb_mkdir_all(const char *pathname, mode_t mode);
+/*
+ * sdb_remove_all:
+ * Recursively deletes the specified path from the filesystem.
+ *
+ * Returns:
+ *  - 0 on success
+ *  - a negative value else
+ */
+int
+sdb_remove_all(const char *pathname);
+
 #ifdef __cplusplus
 } /* extern "C" */
 #endif
diff --git a/src/utils/os.c b/src/utils/os.c
index 08cb188d172355126d16f80ed4ae096aad47ff06..7858825e1c889ad040b028284b9c4725c29a8068 100644 (file)
--- a/src/utils/os.c
+++ b/src/utils/os.c
 #include <sys/types.h>
 #include <sys/stat.h>
+#include <dirent.h>
+
 #include <stdlib.h>
+#include <stdio.h>
 #include <string.h>
 #include <unistd.h>
        return status;
 } /* sdb_mkdir_all */
+int
+sdb_remove_all(const char *pathname)
+{
+       struct stat st;
+
+       if ((! pathname) || (! *pathname)) {
+               errno = EINVAL;
+               return -1;
+       }
+
+       memset(&st, 0, sizeof(st));
+       if (stat(pathname, &st))
+               return -1;
+
+       if (S_ISDIR(st.st_mode)) {
+               DIR *d = opendir(pathname);
+
+               if (! d)
+                       return -1;
+
+               while (42) {
+                       struct dirent de;
+                       struct dirent *res = NULL;
+
+                       char filename[strlen(pathname) + sizeof(de.d_name) + 2];
+
+                       memset(&de, 0, sizeof(de));
+                       if (readdir_r(d, &de, &res)) {
+                               closedir(d);
+                               return -1;
+                       }
+
+                       if (! res)
+                               break;
+
+                       if ((de.d_name[0] == '.') && ((de.d_name[1] == '\0')
+                                               || ((de.d_name[1] == '.')&& (de.d_name[2] == '\0'))))
+                               continue;
+
+                       snprintf(filename, sizeof(filename),
+                                       "%s/%s", pathname, de.d_name);
+                       if (sdb_remove_all(filename)) {
+                               closedir(d);
+                               return -1;
+                       }
+               }
+               closedir(d);
+       }
+       return remove(pathname);
+} /* sdb_remove_all */
+
 /* vim: set tw=78 sw=4 ts=4 noexpandtab : */
![[tokkee]](http://tokkee.org/images/avatar.png)
