/*
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;
}