/* Copyright (C) 2011 ABRT Team Copyright (C) 2011 RedHat inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "internal_libreport.h" struct dump_dir *steal_directory(const char *base_dir, const char *dump_dir_name) { const char *base_name = strrchr(dump_dir_name, '/'); if (base_name) { if (base_name[1] == '\0') { /* Drats. It has trailing slash(es) */ /* Skip all trailing slashes */ while (base_name > dump_dir_name && *--base_name == '/') continue; /* Find previous one */ for (;;) { if (*base_name == '/') break; base_name--; if (base_name < dump_dir_name) /* It has ONLY trailing slash(es) */ break; } } base_name++; } else base_name = dump_dir_name; struct dump_dir *dd_dst; unsigned count = 100; char *dst_dir_name = concat_path_file(base_dir, base_name); while (1) { dd_dst = dd_create(dst_dir_name, (uid_t)-1, DEFAULT_DUMP_DIR_MODE); free(dst_dir_name); if (dd_dst) break; if (--count == 0) { error_msg("Can't create new dump dir in '%s'", base_dir); return NULL; } struct timeval tv; gettimeofday(&tv, NULL); dst_dir_name = xasprintf("%s/%s.%u", base_dir, base_name, (int)tv.tv_usec); } log_notice("Creating copy in '%s'", dd_dst->dd_dirname); if (copy_file_recursive(dump_dir_name, dd_dst->dd_dirname) < 0) { /* error. copy_file_recursive already emitted error message */ /* Don't leave half-copied dir lying around */ dd_delete(dd_dst); return NULL; } return dd_dst; } struct dump_dir *open_directory_for_writing( const char *dump_dir_name, bool (*ask)(const char *, const char *)) { struct dump_dir *dd = dd_opendir(dump_dir_name, DD_OPEN_READONLY); if (!dd) xfunc_die(); /* error msg was already logged */ if (dd->locked) return dd; log_warning("'%s' is not writable", dump_dir_name); dd_close(dd); char *spooldir = concat_path_file(g_get_user_cache_dir(), "abrt/spool"); if (ask && !ask(spooldir, dump_dir_name)) return NULL; dd = steal_directory(spooldir, dump_dir_name); free(spooldir); if (!dd) return NULL; bool dd_was_cwd = false; { char old_cwd[PATH_MAX + 1]; /* must get CWD before deleting */ if (getcwd(old_cwd, sizeof(old_cwd))) dd_was_cwd = strcmp(old_cwd, dump_dir_name) == 0; else perror_msg("getcwd()"); } /* Delete old dir and switch to new one. * Don't want to keep new dd open across deletion, * therefore it's a bit more complicated. */ delete_dump_dir_possibly_using_abrtd(dump_dir_name); char *new_name = xstrdup(dd->dd_dirname); dd_close(dd); dd = dd_opendir(new_name, 0); free(new_name); if (!dd) xfunc_die(); /* error msg was already logged */ /* Update CWD to the new dump dir path if CWD is the stolen dump dir */ /* Nonexisting CWD breaks lot of things (i.e. gnome-open can't open URL)*/ if (dd_was_cwd && chdir(dd->dd_dirname)) perror_msg("chdir()"); return dd; }