|
|
4b6aa8 |
From f9ff58e8cde1d3b6f5eb5307e2019d54c5e28487 Mon Sep 17 00:00:00 2001
|
|
|
4b6aa8 |
From: Matej Habrnal <mhabrnal@redhat.com>
|
|
|
4b6aa8 |
Date: Thu, 11 Feb 2016 17:29:50 +0100
|
|
|
4b6aa8 |
Subject: [PATCH] dd: add a function for compressing dumpdirs
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
Introduce dd_create_archive() function.
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
I added the argument 'flags' in order to avoid the need to introduce
|
|
|
4b6aa8 |
dd_create_archive_ext() function in the future. I am sure will use this
|
|
|
4b6aa8 |
flag in the future (e.g. Encrypted).
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
Signed-off-by: Jakub Filak <jfilak@redhat.com>
|
|
|
4b6aa8 |
Signed-off-by: Matej Habrnal <mhabrnal@redhat.com>
|
|
|
4b6aa8 |
---
|
|
|
4b6aa8 |
po/POTFILES.in | 1 +
|
|
|
4b6aa8 |
src/include/dump_dir.h | 26 +++++
|
|
|
4b6aa8 |
src/lib/Makefile.am | 1 +
|
|
|
4b6aa8 |
src/lib/dump_dir.c | 117 ++++++++++++++++++++
|
|
|
4b6aa8 |
src/plugins/Makefile.am | 2 -
|
|
|
4b6aa8 |
tests/dump_dir.at | 275 ++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
4b6aa8 |
6 files changed, 420 insertions(+), 2 deletions(-)
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
diff --git a/po/POTFILES.in b/po/POTFILES.in
|
|
|
4b6aa8 |
index 00046e2..30c9cb5 100644
|
|
|
4b6aa8 |
--- a/po/POTFILES.in
|
|
|
4b6aa8 |
+++ b/po/POTFILES.in
|
|
|
4b6aa8 |
@@ -18,6 +18,7 @@ src/lib/abrt_sock.c
|
|
|
4b6aa8 |
src/lib/client.c
|
|
|
4b6aa8 |
src/lib/create_dump_dir.c
|
|
|
4b6aa8 |
src/lib/curl.c
|
|
|
4b6aa8 |
+src/lib/dump_dir.c
|
|
|
4b6aa8 |
src/lib/event_config.c
|
|
|
4b6aa8 |
src/lib/ureport.c
|
|
|
4b6aa8 |
src/lib/make_descr.c
|
|
|
4b6aa8 |
diff --git a/src/include/dump_dir.h b/src/include/dump_dir.h
|
|
|
4b6aa8 |
index 07b119a..092ddeb 100644
|
|
|
4b6aa8 |
--- a/src/include/dump_dir.h
|
|
|
4b6aa8 |
+++ b/src/include/dump_dir.h
|
|
|
4b6aa8 |
@@ -21,6 +21,9 @@
|
|
|
4b6aa8 |
#ifndef LIBREPORT_DUMP_DIR_H_
|
|
|
4b6aa8 |
#define LIBREPORT_DUMP_DIR_H_
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
+/* For const_string_vector_const_ptr_t */
|
|
|
4b6aa8 |
+#include "libreport_types.h"
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
/* For DIR */
|
|
|
4b6aa8 |
#include <sys/types.h>
|
|
|
4b6aa8 |
#include <dirent.h>
|
|
|
4b6aa8 |
@@ -178,6 +181,29 @@ int fdump_dir_stat_for_uid(int dir_fd, uid_t uid);
|
|
|
4b6aa8 |
*/
|
|
|
4b6aa8 |
int dd_mark_as_notreportable(struct dump_dir *dd, const char *reason);
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
+/* Creates a new archive from the dump directory contents
|
|
|
4b6aa8 |
+ *
|
|
|
4b6aa8 |
+ * The dd argument must be opened for reading.
|
|
|
4b6aa8 |
+ *
|
|
|
4b6aa8 |
+ * The archive_name must not exist. The file will be created with 0600 mode.
|
|
|
4b6aa8 |
+ *
|
|
|
4b6aa8 |
+ * The archive type is deduced from archive_name suffix. The supported archive
|
|
|
4b6aa8 |
+ * suffixes are the following:
|
|
|
4b6aa8 |
+ * - '.tag.gz' (note: the implementation uses child gzip process)
|
|
|
4b6aa8 |
+ *
|
|
|
4b6aa8 |
+ * The archive will include only the files that are not in the exclude_elements
|
|
|
4b6aa8 |
+ * list. See get_global_always_excluded_elements().
|
|
|
4b6aa8 |
+ *
|
|
|
4b6aa8 |
+ * The argument "flags" is currently unused.
|
|
|
4b6aa8 |
+ *
|
|
|
4b6aa8 |
+ * @return 0 on success; otherwise non-0 value. -ENOSYS if archive type is not
|
|
|
4b6aa8 |
+ * supported. -EEXIST if the archive file already exists. -ECHILD if child
|
|
|
4b6aa8 |
+ * process fails. Other negative values can be converted to errno values by
|
|
|
4b6aa8 |
+ * turning them positive.
|
|
|
4b6aa8 |
+ */
|
|
|
4b6aa8 |
+int dd_create_archive(struct dump_dir *dd, const char *archive_name,
|
|
|
4b6aa8 |
+ const_string_vector_const_ptr_t exclude_elements, int flags);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
#ifdef __cplusplus
|
|
|
4b6aa8 |
}
|
|
|
4b6aa8 |
#endif
|
|
|
4b6aa8 |
diff --git a/src/lib/Makefile.am b/src/lib/Makefile.am
|
|
|
4b6aa8 |
index f9ea602..50142f7 100644
|
|
|
4b6aa8 |
--- a/src/lib/Makefile.am
|
|
|
4b6aa8 |
+++ b/src/lib/Makefile.am
|
|
|
4b6aa8 |
@@ -79,6 +79,7 @@ libreport_la_CPPFLAGS = \
|
|
|
4b6aa8 |
$(AUGEAS_CFLAGS) \
|
|
|
4b6aa8 |
-D_GNU_SOURCE
|
|
|
4b6aa8 |
libreport_la_LDFLAGS = \
|
|
|
4b6aa8 |
+ -ltar \
|
|
|
4b6aa8 |
-version-info 0:1:0
|
|
|
4b6aa8 |
libreport_la_LIBADD = \
|
|
|
4b6aa8 |
$(JSON_C_LIBS) \
|
|
|
4b6aa8 |
diff --git a/src/lib/dump_dir.c b/src/lib/dump_dir.c
|
|
|
4b6aa8 |
index 9096853..a5cd93e 100644
|
|
|
4b6aa8 |
--- a/src/lib/dump_dir.c
|
|
|
4b6aa8 |
+++ b/src/lib/dump_dir.c
|
|
|
4b6aa8 |
@@ -17,6 +17,7 @@
|
|
|
4b6aa8 |
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
4b6aa8 |
*/
|
|
|
4b6aa8 |
#include <sys/utsname.h>
|
|
|
4b6aa8 |
+#include <libtar.h>
|
|
|
4b6aa8 |
#include "internal_libreport.h"
|
|
|
4b6aa8 |
|
|
|
4b6aa8 |
// Locking logic:
|
|
|
4b6aa8 |
@@ -1475,3 +1476,119 @@ int dd_mark_as_notreportable(struct dump_dir *dd, const char *reason)
|
|
|
4b6aa8 |
dd_save_text(dd, FILENAME_NOT_REPORTABLE, reason);
|
|
|
4b6aa8 |
return 0;
|
|
|
4b6aa8 |
}
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+/* flags - for future needs */
|
|
|
4b6aa8 |
+int dd_create_archive(struct dump_dir *dd, const char *archive_name,
|
|
|
4b6aa8 |
+ const_string_vector_const_ptr_t exclude_elements, int flags)
|
|
|
4b6aa8 |
+{
|
|
|
4b6aa8 |
+ if (suffixcmp(archive_name, ".tar.gz") != 0)
|
|
|
4b6aa8 |
+ return -ENOSYS;
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ int result = 0;
|
|
|
4b6aa8 |
+ pid_t child;
|
|
|
4b6aa8 |
+ TAR* tar = NULL;
|
|
|
4b6aa8 |
+ int pipe_from_parent_to_child[2];
|
|
|
4b6aa8 |
+ xpipe(pipe_from_parent_to_child);
|
|
|
4b6aa8 |
+ child = fork();
|
|
|
4b6aa8 |
+ if (child < 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ result = -errno;
|
|
|
4b6aa8 |
+ /* Don't die, let the caller to execute his clean-up code. */
|
|
|
4b6aa8 |
+ perror_msg("vfork");
|
|
|
4b6aa8 |
+ return result;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+ if (child == 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ /* child */
|
|
|
4b6aa8 |
+ close(pipe_from_parent_to_child[1]);
|
|
|
4b6aa8 |
+ xmove_fd(pipe_from_parent_to_child[0], 0);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ int fd = open(archive_name, O_WRONLY | O_CREAT | O_EXCL, 0600);
|
|
|
4b6aa8 |
+ if (fd < 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ /* This r might interfer with exit status of gzip, but it is
|
|
|
4b6aa8 |
+ * very unlikely (man 1 gzip):
|
|
|
4b6aa8 |
+ * Exit status is normally 0; if an error occurs, exit status is
|
|
|
4b6aa8 |
+ * 1. If a warning occurs, exit status is 2.
|
|
|
4b6aa8 |
+ */
|
|
|
4b6aa8 |
+ result = errno == EEXIST ? 100 : 3;
|
|
|
4b6aa8 |
+ perror_msg("Can't open '%s'", archive_name);
|
|
|
4b6aa8 |
+ exit(result);
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ xmove_fd(fd, 1);
|
|
|
4b6aa8 |
+ execlp("gzip", "gzip", NULL);
|
|
|
4b6aa8 |
+ perror_msg_and_die("Can't execute '%s'", "gzip");
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+ close(pipe_from_parent_to_child[0]);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* If child died (say, in xopen), then parent might get SIGPIPE.
|
|
|
4b6aa8 |
+ * We want to properly unlock dd, therefore we must not die on SIGPIPE:
|
|
|
4b6aa8 |
+ */
|
|
|
4b6aa8 |
+ sighandler_t old_handler = signal(SIGPIPE, SIG_IGN);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* Create tar writer object */
|
|
|
4b6aa8 |
+ if (tar_fdopen(&tar, pipe_from_parent_to_child[1], (char *)archive_name,
|
|
|
4b6aa8 |
+ /*fileops:(standard)*/ NULL, O_WRONLY | O_CREAT, 0644, TAR_GNU) != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ result = -errno;
|
|
|
4b6aa8 |
+ log_warning(_("Failed to open TAR writer"));
|
|
|
4b6aa8 |
+ goto finito;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* Write data to the tarball */
|
|
|
4b6aa8 |
+ dd_init_next_file(dd);
|
|
|
4b6aa8 |
+ char *short_name, *full_name;
|
|
|
4b6aa8 |
+ while (dd_get_next_file(dd, &short_name, &full_name))
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ if (!(exclude_elements && is_in_string_list(short_name, exclude_elements)))
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ if (tar_append_file(tar, full_name, short_name))
|
|
|
4b6aa8 |
+ result = -errno;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ free(short_name);
|
|
|
4b6aa8 |
+ free(full_name);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (result != 0)
|
|
|
4b6aa8 |
+ goto finito;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* Close tar writer... */
|
|
|
4b6aa8 |
+ if (tar_append_eof(tar) != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ result = -errno;
|
|
|
4b6aa8 |
+ log_warning(_("Failed to finalize TAR archive"));
|
|
|
4b6aa8 |
+ goto finito;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+finito:
|
|
|
4b6aa8 |
+ signal(SIGPIPE, old_handler);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (tar != NULL && tar_close(tar) != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ result = -errno;
|
|
|
4b6aa8 |
+ log_warning(_("Failed to close TAR writer"));
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* ...and check that gzip child finished successfully */
|
|
|
4b6aa8 |
+ int status;
|
|
|
4b6aa8 |
+ safe_waitpid(child, &status, 0);
|
|
|
4b6aa8 |
+ if (status != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ result = -ECHILD;
|
|
|
4b6aa8 |
+ if (WIFSIGNALED(status))
|
|
|
4b6aa8 |
+ log_warning(_("gzip killed with signal %d"), WTERMSIG(status));
|
|
|
4b6aa8 |
+ else if (WIFEXITED(status))
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ if (WEXITSTATUS(status) == 100)
|
|
|
4b6aa8 |
+ result = -EEXIST;
|
|
|
4b6aa8 |
+ else
|
|
|
4b6aa8 |
+ log_warning(_("gzip exited with %d"), WEXITSTATUS(status));
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+ else
|
|
|
4b6aa8 |
+ log_warning(_("gzip process failed"));
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ return result;
|
|
|
4b6aa8 |
+}
|
|
|
4b6aa8 |
diff --git a/src/plugins/Makefile.am b/src/plugins/Makefile.am
|
|
|
4b6aa8 |
index 7ec08d7..d5d75b6 100644
|
|
|
4b6aa8 |
--- a/src/plugins/Makefile.am
|
|
|
4b6aa8 |
+++ b/src/plugins/Makefile.am
|
|
|
4b6aa8 |
@@ -145,7 +145,6 @@ reporter_rhtsupport_CPPFLAGS = \
|
|
|
4b6aa8 |
$(LIBREPORT_CFLAGS) \
|
|
|
4b6aa8 |
$(LIBXML_CFLAGS) \
|
|
|
4b6aa8 |
-D_GNU_SOURCE
|
|
|
4b6aa8 |
-reporter_rhtsupport_LDFLAGS = -ltar
|
|
|
4b6aa8 |
reporter_rhtsupport_LDADD = \
|
|
|
4b6aa8 |
$(GLIB_LIBS) \
|
|
|
4b6aa8 |
$(LIBXML_LIBS) \
|
|
|
4b6aa8 |
@@ -168,7 +167,6 @@ reporter_upload_CPPFLAGS = \
|
|
|
4b6aa8 |
$(CURL_CFLAGS) \
|
|
|
4b6aa8 |
$(LIBREPORT_CFLAGS) \
|
|
|
4b6aa8 |
-D_GNU_SOURCE
|
|
|
4b6aa8 |
-reporter_upload_LDFLAGS = -ltar
|
|
|
4b6aa8 |
reporter_upload_LDADD = \
|
|
|
4b6aa8 |
$(GLIB_LIBS) \
|
|
|
4b6aa8 |
../lib/libreport-web.la \
|
|
|
4b6aa8 |
diff --git a/tests/dump_dir.at b/tests/dump_dir.at
|
|
|
4b6aa8 |
index a579243..fb8c7ce 100644
|
|
|
4b6aa8 |
--- a/tests/dump_dir.at
|
|
|
4b6aa8 |
+++ b/tests/dump_dir.at
|
|
|
4b6aa8 |
@@ -47,3 +47,278 @@ int main(void)
|
|
|
4b6aa8 |
return 0;
|
|
|
4b6aa8 |
}
|
|
|
4b6aa8 |
]])
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+## ----------------- ##
|
|
|
4b6aa8 |
+## dd_create_archive ##
|
|
|
4b6aa8 |
+## ----------------- ##
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+AT_TESTFUN([dd_create_archive],
|
|
|
4b6aa8 |
+[[
|
|
|
4b6aa8 |
+#include "internal_libreport.h"
|
|
|
4b6aa8 |
+#include <libtar.h>
|
|
|
4b6aa8 |
+#include <assert.h>
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+void verify_archive(struct dump_dir *dd, const char *file_name,
|
|
|
4b6aa8 |
+ const_string_vector_const_ptr_t included_files,
|
|
|
4b6aa8 |
+ const_string_vector_const_ptr_t excluded_files)
|
|
|
4b6aa8 |
+{
|
|
|
4b6aa8 |
+ unsigned c = 0;
|
|
|
4b6aa8 |
+ for (const_string_vector_const_ptr_t i = included_files; i && *i; ++i)
|
|
|
4b6aa8 |
+ ++c;
|
|
|
4b6aa8 |
+ int *check_array = xzalloc(c * sizeof(int));
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ int pipe_from_parent_to_child[2];
|
|
|
4b6aa8 |
+ xpipe(pipe_from_parent_to_child);
|
|
|
4b6aa8 |
+ pid_t child = fork();
|
|
|
4b6aa8 |
+ if (child < 0)
|
|
|
4b6aa8 |
+ perror_msg_and_die("vfork");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (child == 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ /* child */
|
|
|
4b6aa8 |
+ close(pipe_from_parent_to_child[0]);
|
|
|
4b6aa8 |
+ xmove_fd(xopen(file_name, O_RDONLY), 0);
|
|
|
4b6aa8 |
+ xmove_fd(pipe_from_parent_to_child[1], 1);
|
|
|
4b6aa8 |
+ execlp("gzip", "gzip", "-d", NULL);
|
|
|
4b6aa8 |
+ perror_msg_and_die("Can't execute '%s'", "gzip");
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+ close(pipe_from_parent_to_child[1]);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* If child died (say, in xopen), then parent might get SIGPIPE.
|
|
|
4b6aa8 |
+ * We want to properly unlock dd, therefore we must not die on SIGPIPE:
|
|
|
4b6aa8 |
+ */
|
|
|
4b6aa8 |
+ signal(SIGPIPE, SIG_IGN);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ TAR* tar = NULL;
|
|
|
4b6aa8 |
+ /* Create tar writer object */
|
|
|
4b6aa8 |
+ if (tar_fdopen(&tar, pipe_from_parent_to_child[0], (char *)file_name,
|
|
|
4b6aa8 |
+ /*fileops:(standard)*/ NULL, O_RDONLY, 0644, TAR_GNU) != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "Failed to open the pipe to gzip for archive: '%s'\n", file_name);
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ int r = 0;
|
|
|
4b6aa8 |
+ const char *real_file = "/tmp/libreport-attest-extracted";
|
|
|
4b6aa8 |
+ while ((r = th_read(tar)) == 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ char *path = th_get_pathname(tar);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (!TH_ISREG(tar))
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "Not regular file: '%s', found in archive: '%s'\n", path, file_name);
|
|
|
4b6aa8 |
+ continue;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ const_string_vector_const_ptr_t i = included_files;
|
|
|
4b6aa8 |
+ for (c = 0; i && *i; ++i, ++c)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ if (strcmp(*i, path) == 0)
|
|
|
4b6aa8 |
+ break;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (i && *i != NULL)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ printf("Included file: '%s', found in archive '%s'\n", path, file_name);
|
|
|
4b6aa8 |
+ check_array[c] += 1;
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ unlink(real_file);
|
|
|
4b6aa8 |
+ if(tar_extract_regfile(tar, xstrdup(real_file)) != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "TAR failed to extract '%s' to '%s': %s\n", path, real_file, strerror(errno));
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ char *original = dd_load_text(dd, path);
|
|
|
4b6aa8 |
+ assert(original != NULL);
|
|
|
4b6aa8 |
+ assert(original[0] != '\0');
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ char *extracted = xmalloc_xopen_read_close("/tmp/libreport-attest-extracted", NULL);
|
|
|
4b6aa8 |
+ assert(extracted != NULL);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (strcmp(extracted, original) != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "Invalid file contents: '%s'\nExp: '%s'\nGot: '%s'\n", path, original, extracted);
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ free(original);
|
|
|
4b6aa8 |
+ free(extracted);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ continue;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ i = excluded_files;
|
|
|
4b6aa8 |
+ for (; i && *i; ++i)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ if (strcmp(*i, path) == 0)
|
|
|
4b6aa8 |
+ break;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (i && *i != NULL)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "Excluded file: '%s', found in archive '%s'\n", path, file_name);
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ fprintf(stderr, "Uncategorized file: '%s', found in archive '%s'\n", path, file_name);
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (r != 1)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "th_read() failed: %s\n", strerror(errno));
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ tar_close(tar);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ int status;
|
|
|
4b6aa8 |
+ safe_waitpid(child, &status, 0);
|
|
|
4b6aa8 |
+ if (status != 0)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "gzip status code '%d'\n", status);
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ int err = 0;
|
|
|
4b6aa8 |
+ const_string_vector_const_ptr_t i = included_files;
|
|
|
4b6aa8 |
+ for (c = 0; i && *i; ++i, ++c)
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ switch (check_array[c])
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ case 0:
|
|
|
4b6aa8 |
+ fprintf(stderr, "Not found included file: '%s', in archive: %s\n", *i, file_name);
|
|
|
4b6aa8 |
+ ++err;
|
|
|
4b6aa8 |
+ break;
|
|
|
4b6aa8 |
+ case 1:
|
|
|
4b6aa8 |
+ fprintf(stdout, "Found included file: '%s', in archive: %s\n", *i, file_name);
|
|
|
4b6aa8 |
+ break;
|
|
|
4b6aa8 |
+ default:
|
|
|
4b6aa8 |
+ fprintf(stderr, "%d occurrences of included file: '%s', in archive: %s\n", check_array[c], *i, file_name);
|
|
|
4b6aa8 |
+ ++err;
|
|
|
4b6aa8 |
+ break;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (err)
|
|
|
4b6aa8 |
+ abort();
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ return;
|
|
|
4b6aa8 |
+}
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+int main(void)
|
|
|
4b6aa8 |
+{
|
|
|
4b6aa8 |
+ g_verbose = 3;
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ char template[] = "/tmp/XXXXXX";
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ if (mkdtemp(template) == NULL) {
|
|
|
4b6aa8 |
+ perror("mkdtemp()");
|
|
|
4b6aa8 |
+ return EXIT_FAILURE;
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ printf("Dump dir path: %s\n", template);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ struct dump_dir *dd = dd_create(template, (uid_t)-1, 0640);
|
|
|
4b6aa8 |
+ assert(dd != NULL || !"Cannot create new dump directory");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+#define COMMON_FILES "time", "last_occurrence", "uid", "kernel", \
|
|
|
4b6aa8 |
+ "architecture", "hostname", "os_info", "os_release", \
|
|
|
4b6aa8 |
+ "type", "count", "component", "program_log"
|
|
|
4b6aa8 |
+#define SENSITIVE_FILES "environ", "backtrace", "secret_file", "private_file", \
|
|
|
4b6aa8 |
+ "useless_file"
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ dd_create_basic_files(dd, geteuid(), NULL);
|
|
|
4b6aa8 |
+ dd_save_text(dd, FILENAME_TYPE, "attest");
|
|
|
4b6aa8 |
+ dd_save_text(dd, FILENAME_COUNT, "1");
|
|
|
4b6aa8 |
+ dd_save_text(dd, FILENAME_COMPONENT, "libreport-attest");
|
|
|
4b6aa8 |
+ dd_save_text(dd, "program_log", "Something very important!");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ const gchar *excluded_files[] = {
|
|
|
4b6aa8 |
+ SENSITIVE_FILES,
|
|
|
4b6aa8 |
+ NULL,
|
|
|
4b6aa8 |
+ };
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ for (const gchar **iter = excluded_files; *iter; ++iter)
|
|
|
4b6aa8 |
+ dd_save_text(dd, *iter, *iter);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* Un-supported archive type */
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "TEST-CASE: Un-supported type\n");
|
|
|
4b6aa8 |
+ fprintf(stdout, "TEST-CASE: Un-supported type\n");
|
|
|
4b6aa8 |
+ const int r = dd_create_archive(dd, "/tmp/libreport-attest.omg", NULL, 0);
|
|
|
4b6aa8 |
+ printf("dd_create_archive() == %d\n", r);
|
|
|
4b6aa8 |
+ assert(r == -ENOSYS || !"Not supported");
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ /* File already exists. */
|
|
|
4b6aa8 |
+ dd_close(dd);
|
|
|
4b6aa8 |
+ dd = dd_opendir(template, DD_OPEN_READONLY);
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "TEST-CASE: File exists\n");
|
|
|
4b6aa8 |
+ fprintf(stdout, "TEST-CASE: File exists\n");
|
|
|
4b6aa8 |
+ char file_contents[] = "Non emtpy file";
|
|
|
4b6aa8 |
+ const char *file_name = "/tmp/libreport-attest.tar.gz";
|
|
|
4b6aa8 |
+ FILE *test_file = fopen(file_name, "w");
|
|
|
4b6aa8 |
+ assert(test_file != NULL);
|
|
|
4b6aa8 |
+ assert(fprintf(test_file, "%s", file_contents) == strlen(file_contents));
|
|
|
4b6aa8 |
+ fclose(test_file);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ assert(dd_create_archive(dd, file_name, NULL, 0) == -EEXIST || !"Exists");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ char *canary = xmalloc_xopen_read_close(file_name, NULL);
|
|
|
4b6aa8 |
+ assert(canary != NULL);
|
|
|
4b6aa8 |
+ assert(strcmp(canary, file_contents) == 0);
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ dd_close(dd);
|
|
|
4b6aa8 |
+ dd = dd_opendir(template, DD_OPEN_READONLY);
|
|
|
4b6aa8 |
+ /* All elements */
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "TEST-CASE: Compress all elements\n");
|
|
|
4b6aa8 |
+ fprintf(stdout, "TEST-CASE: Compress all elements\n");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ const gchar *included_files[] = {
|
|
|
4b6aa8 |
+ COMMON_FILES,
|
|
|
4b6aa8 |
+ SENSITIVE_FILES,
|
|
|
4b6aa8 |
+ NULL,
|
|
|
4b6aa8 |
+ };
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ const char *file_name = "/tmp/libreport-attest-all.tar.gz";
|
|
|
4b6aa8 |
+ unlink(file_name);
|
|
|
4b6aa8 |
+ assert(dd_create_archive(dd, file_name, NULL, 0) == 0 || !"All elements");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ verify_archive(dd, file_name, included_files, NULL);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ unlink(file_name);
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ dd_close(dd);
|
|
|
4b6aa8 |
+ dd = dd_opendir(template, DD_OPEN_READONLY);
|
|
|
4b6aa8 |
+ /* Excluded elements */
|
|
|
4b6aa8 |
+ {
|
|
|
4b6aa8 |
+ fprintf(stderr, "TEST-CASE: Exclude elements\n");
|
|
|
4b6aa8 |
+ fprintf(stdout, "TEST-CASE: Exclude elements\n");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ const char *included_files[] = {
|
|
|
4b6aa8 |
+ COMMON_FILES,
|
|
|
4b6aa8 |
+ NULL,
|
|
|
4b6aa8 |
+ };
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ const char *file_name = "/tmp/libreport-attest-excluded.tar.gz";
|
|
|
4b6aa8 |
+ unlink(file_name);
|
|
|
4b6aa8 |
+ assert(dd_create_archive(dd, file_name, excluded_files, 0) == 0 || !"Excluded elements");
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ verify_archive(dd, file_name, included_files, excluded_files);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ unlink(file_name);
|
|
|
4b6aa8 |
+ }
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ dd_close(dd);
|
|
|
4b6aa8 |
+ dd = dd_opendir(template, DD_OPEN_READONLY);
|
|
|
4b6aa8 |
+ assert(dd_delete(dd) == 0);
|
|
|
4b6aa8 |
+
|
|
|
4b6aa8 |
+ return 0;
|
|
|
4b6aa8 |
+}
|
|
|
4b6aa8 |
+]])
|
|
|
4b6aa8 |
--
|
|
|
4b6aa8 |
1.8.3.1
|
|
|
4b6aa8 |
|