/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* camel-maildir-folder.c : camel-folder subclass for maildir folders */

/* 
 *
 * Copyright (C) 1999 Bertrand Guiheneuf <Bertrand.Guiheneuf@inria.fr> .
 *
 * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 * USA
 */

/*
 * AUTHORS : Jukka Zitting 
 *  
 */


#include <config.h> 
#include <sys/stat.h> 
#include <sys/param.h> 
#include <unistd.h>
#include <sys/types.h>
#include <fcntl.h>
#include <dirent.h>
#include <stdio.h>
#include <errno.h>
#include <time.h>
#include <string.h>
#include "camel-maildir-folder.h"
#include "camel-maildir-store.h"
#include "camel-stream-fs.h"
#include "camel-log.h"

static CamelFolderClass *parent_class=NULL;

/* Returns the class for a CamelMaildirFolder */
#define CMAILDIRF_CLASS(so) CAMEL_MAILDIR_FOLDER_CLASS (GTK_OBJECT(so)->klass)
#define CF_CLASS(so) CAMEL_FOLDER_CLASS (GTK_OBJECT(so)->klass)
#define CMAILDIRS_CLASS(so) CAMEL_STORE_CLASS (GTK_OBJECT(so)->klass)

static void _init_with_store (CamelFolder *folder, CamelStore *parent_store, CamelException *ex);
static void _set_name (CamelFolder *folder, const gchar *name, CamelException *ex);
static gboolean _exists (CamelFolder *folder, CamelException *ex);
static gboolean _create (CamelFolder *folder, CamelException *ex);
static gboolean _delete (CamelFolder *folder, gboolean recurse, CamelException *ex);
static gboolean _delete_messages (CamelFolder *folder, CamelException *ex);
static CamelMimeMessage *_get_message (CamelFolder *folder, gint number, CamelException *ex);
static gint _get_message_count (CamelFolder *folder, CamelException *ex);
static void _expunge (CamelFolder *folder, CamelException *ex);
static GList *_list_subfolders (CamelFolder *folder, CamelException *ex);

/* fs utility functions */
static DIR * _xopendir (const gchar *path);
static gboolean _xstat (const gchar *path, struct stat *buf);
static gboolean _xmkdir (const gchar *path);
static gboolean _xrename (const gchar *from, const gchar *to);
static gboolean _xunlink (const gchar *path);
static gboolean _xrmdir (const gchar *path);
/* ** */

static void
camel_maildir_folder_class_init (CamelMaildirFolderClass *camel_maildir_folder_class)
{
	CamelFolderClass *camel_folder_class =
		CAMEL_FOLDER_CLASS (camel_maildir_folder_class);

	parent_class = gtk_type_class (camel_folder_get_type ());

	/* virtual method definition */
	/* virtual method overload */
	camel_folder_class->init_with_store   = _init_with_store;
	camel_folder_class->set_name          = _set_name;
	camel_folder_class->exists            = _exists;
	camel_folder_class->create            = _create;
	camel_folder_class->delete            = _delete;
	camel_folder_class->delete_messages   = _delete_messages;
	camel_folder_class->expunge           = _expunge;
	camel_folder_class->get_message       = _get_message;
	camel_folder_class->get_message_count = _get_message_count;
	camel_folder_class->list_subfolders   = _list_subfolders;
}

GtkType
camel_maildir_folder_get_type (void)
{
	static GtkType camel_maildir_folder_type = 0;
	
	if (!camel_maildir_folder_type)	{
		GtkTypeInfo camel_maildir_folder_info =	
		{
			"CamelMaildirFolder",
			sizeof (CamelMaildirFolder),
			sizeof (CamelMaildirFolderClass),
			(GtkClassInitFunc) camel_maildir_folder_class_init,
			(GtkObjectInitFunc) NULL,
				/* reserved_1 */ NULL,
				/* reserved_2 */ NULL,
			(GtkClassInitFunc) NULL,
		};
		
		camel_maildir_folder_type =
			gtk_type_unique (CAMEL_FOLDER_TYPE, &camel_maildir_folder_info);
	}
	
	return camel_maildir_folder_type;
}






/**
 * CamelMaildirFolder::init_with_store: initializes the folder object
 * @folder:       folder object to initialize
 * @parent_store: parent store object of the folder
 *
 * Simply tells that the folder can contain messages but not subfolders.
 * Perhaps we'll later implement subfolders too...
 */
static void 
_init_with_store (CamelFolder *folder, CamelStore *parent_store, CamelException *ex)
{
	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::init_with_store\n");
	g_assert (folder);
	g_assert (parent_store);
	
	/* call parent method */
	parent_class->init_with_store (folder, parent_store, ex);
	
	folder->can_hold_messages = TRUE;
	folder->can_hold_folders = TRUE;
	folder->has_summary_capability = FALSE;

	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::init_with_store\n");
}

/**
 * CamelMaildirFolder::set_name: sets the name of the folder
 * @folder: folder object
 * @name:   name of the folder
 *
 * Sets the name of the folder object. The existence of a folder with
 * the given name is not checked in this function.
 */
static void
_set_name (CamelFolder *folder, const gchar *name, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder;
	CamelMaildirStore *maildir_store;
	
	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::set_name\n");
	g_assert (folder);
	g_assert (name);
	g_assert (folder->parent_store);

	maildir_folder = CAMEL_MAILDIR_FOLDER (folder);
	maildir_store = CAMEL_MAILDIR_STORE (folder->parent_store);

	/* call default implementation */
	parent_class->set_name (folder, name, ex);
	
	if (maildir_folder->directory_path)
		g_free (maildir_folder->directory_path);

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::set_name full_name is %s\n", folder->full_name);
	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::set_name toplevel_dir is %s\n", maildir_store->toplevel_dir);
	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::set_name separator is %c\n", camel_store_get_separator (folder->parent_store));

	if (folder->full_name && folder->full_name[0])
		maildir_folder->directory_path =
			g_strconcat (maildir_store->toplevel_dir, G_DIR_SEPARATOR_S,
				     folder->full_name, NULL);
	else
		maildir_folder->directory_path = g_strdup (maildir_store->toplevel_dir);

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::set_name: name set to %s\n", name);
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::set_name\n");
}

/**
 * CamelMaildirFolder::exists: tests whether the named maildir exists
 * @folder: folder object
 *
 * A created maildir folder object doesn't necessarily exist yet in the
 * filesystem. This function checks whether the maildir exists.
 * The structure of the maildir is stated in the maildir.5 manpage.
 *
 * maildir.5:
 *     A directory in maildir format  has  three  subdirectories,
 *     all on the same filesystem: tmp, new, and cur.
 *
 * Return value: TRUE if the maildir exists, FALSE otherwise
 */
static gboolean
_exists (CamelFolder *folder, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER (folder);
	static const gchar *dir[3] = { "new", "cur", "tmp" };
	gint i;
	struct stat statbuf;
	const gchar *maildir;
	gchar *path;
	gboolean rv = TRUE;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::exists\n");
	g_assert (folder);
	g_return_val_if_fail (maildir_folder->directory_path, FALSE);

	maildir = maildir_folder->directory_path;

	CAMEL_LOG_FULL_DEBUG ("CamelMailFolder::exists: checking maildir %s\n",
			      maildir);

	/* check whether the toplevel directory exists */
	rv = _xstat (maildir, &statbuf) && S_ISDIR (statbuf.st_mode);

	/* check whether the maildir subdirectories exist */
	for (i = 0; rv && i < 3; i++) {
		path = g_strconcat (maildir, G_DIR_SEPARATOR_S, dir[i], NULL);

		rv = _xstat (path, &statbuf) && S_ISDIR (statbuf.st_mode);

		g_free (path);
	}

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::exists: %s\n",
			      (rv) ? "maildir found" : "maildir not found");
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::exists\n");
	return rv;
}

/**
 * CamelMaildirFolder::create: creates the named maildir
 * @folder: folder object
 *
 * A created maildir folder object doesn't necessarily exist yet in the
 * filesystem. This function creates the maildir if it doesn't yet exist.
 * The structure of the maildir is stated in the maildir.5 manpage.
 *
 * maildir.5:
 *     A directory in maildir format  has  three  subdirectories,
 *     all on the same filesystem: tmp, new, and cur.
 *
 * Return value: TRUE if the maildir existed already or was created,
 *               FALSE otherwise
 */
static gboolean
_create (CamelFolder *folder, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER (folder);
	static const gchar *dir[3] = { "new", "cur", "tmp" };
	gint i;
	const gchar *maildir;
	gchar *path;
	gboolean rv = TRUE;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::create\n");
	g_assert (folder);

	/* check whether the maildir already exists */
	if (camel_folder_exists (folder, ex)) return TRUE;

	maildir = maildir_folder->directory_path;

	CAMEL_LOG_FULL_DEBUG ("CamelMailFolder::create: creating maildir %s\n",
			      maildir);

	/* create the toplevel directory */
	rv = _xmkdir (maildir);

	/* create the maildir subdirectories */
	for (i = 0; rv && i < 3; i++) {
		path = g_strconcat (maildir, G_DIR_SEPARATOR_S, dir[i], NULL);

		rv = _xmkdir (path);

		g_free (path);
	}

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::create: %s\n",
			      rv ? "maildir created" : "an error occurred");
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::create\n");
	return rv;
}

/**
 * CamelMaildirFolder::delete: delete the maildir folder
 * @folder: the folder object
 * @recurse:
 *
 * This function empties and deletes the maildir folder. The subdirectories
 * "tmp", "cur", and "new" are removed first and then the toplevel maildir
 * directory is deleted. All files from the directories are deleted as well, 
 * so you should be careful when using this function. If a subdirectory cannot
 * be deleted, then the operation it is stopped. Thus if an error occurs, the
 * maildir directory won't be removed, but it might no longer be a valid maildir.
 */
static gboolean
_delete (CamelFolder *folder, gboolean recurse, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER (folder);
	static const gchar *dir[3] = { "new", "cur", "tmp" };
	gint i;
	const gchar *maildir;
	gchar *path;
	gboolean rv = TRUE;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::create\n");
	g_assert (folder);

	/* check whether the maildir already exists */
	if (!camel_folder_exists (folder, ex)) return TRUE;

	maildir = maildir_folder->directory_path;

	CAMEL_LOG_FULL_DEBUG ("CamelMailFolder::delete: deleting maildir %s\n",
			      maildir);

	/* delete the maildir subdirectories */
	for (i = 0; rv && i < 3; i++) {
		path = g_strconcat (maildir, G_DIR_SEPARATOR_S, dir[i], NULL);

		rv = _xrmdir (path);

		g_free (path);
	}

	/* create the toplevel directory */
	if (rv)
		rv = _xrmdir (maildir);

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::delete: %s\n",
			      rv ? "maildir deleted" : "an error occurred");
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::delete\n");
	return rv;
}

/**
 * CamelMaildirFolder::delete_messages: empty the maildir folder
 * @folder:  the folder object
 *
 * This function empties the maildir folder. All messages from the
 * "cur" subdirectory are deleted. If a message cannot be deleted, then
 * it is just skipped and the rest of the messages are still deleted.
 * Files with names starting with a dot are skipped as described in the
 * maildir.5 manpage.
 *
 * maildir.5:
 *     It is a good idea for readers to skip all filenames in new
 *     and cur starting with a dot. Other than this, readers
 *     should not attempt to parse filenames.
 *
 * Return value: FALSE on error and if some messages could not be deleted.
 *               TRUE otherwise.
 */
static gboolean 
_delete_messages (CamelFolder *folder, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER (folder);
	const gchar *maildir;
	gchar *curdir, *file;
	DIR *dir_handle;
	struct dirent *dir_entry;
	gboolean rv = TRUE;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::delete_messages\n");
	g_assert (folder);

	/* call default implementation */
	parent_class->delete_messages (folder, ex);

	/* Check if the folder didn't exist */
	if (!camel_folder_exists (folder, ex)) return TRUE;

	maildir = maildir_folder->directory_path;

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::delete_messages: "
			      "deleting messages from %s\n", maildir);

	/* delete messages from the maildir subdirectory "cur" */
	curdir = g_strconcat (maildir, G_DIR_SEPARATOR_S, "cur", NULL);

	dir_handle = _xopendir (curdir);
	if (dir_handle) {
		while ((dir_entry = readdir (dir_handle))) {
			if (dir_entry->d_name[0] == '.') continue;
			file = g_strconcat (curdir, G_DIR_SEPARATOR_S,
					    dir_entry->d_name, NULL);

			if (!_xunlink (file)) rv = FALSE;

			g_free (file);
		}
		closedir (dir_handle);
	} else
		rv = FALSE;

	g_free (curdir);
	
	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::delete_messages: %s\n",
			      rv ? "messages deleted" : "an error occurred");
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::delete_messages\n");
	return rv;
}

/**
 * CamelMaildirFolder::get_message: get a message from maildir
 * @folder: the folder object
 * @number: number of the message within the folder
 *
 * Return value: the message, NULL on error
 */
static CamelMimeMessage *
_get_message (CamelFolder *folder, gint number, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER(folder);
	DIR *dir_handle;
	struct dirent *dir_entry;
	CamelStream *stream;
	CamelMimeMessage *message = NULL;
	const gchar *maildir;
	gchar *curdir, *file = NULL;
	gint count = -1;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::get_message\n");
	g_assert(folder);

	/* Check if the folder exists */
	if (!camel_folder_exists (folder, ex)) return NULL;

	maildir = maildir_folder->directory_path;

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::get_message: "
			      "getting message #%d from %s\n", number, maildir);

	/* Count until the desired message is reached */
	curdir = g_strconcat (maildir, G_DIR_SEPARATOR_S, "cur", NULL);
	if ((dir_handle = _xopendir (curdir))) {
		while ((count < number) && (dir_entry = readdir (dir_handle)))
			if (dir_entry->d_name[0] != '.') count++;

		if (count == number)
			file = g_strconcat (curdir, G_DIR_SEPARATOR_S,
					    dir_entry->d_name, NULL);

		closedir (dir_handle);
	}
	g_free (curdir);
	if (!file) return NULL;

	/* Create the message object */
	message = camel_mime_message_new ();
	stream = camel_stream_fs_new_with_name (file, CAMEL_STREAM_FS_READ);

	if (!message || !stream) {
		g_free (file);
		if (stream) gtk_object_unref (GTK_OBJECT (stream));
		if (message) gtk_object_unref (GTK_OBJECT (message));
		return NULL;
	}

	camel_data_wrapper_construct_from_stream (CAMEL_DATA_WRAPPER (message),
						  stream);
	gtk_object_unref (GTK_OBJECT (stream));
	gtk_object_set_data_full (GTK_OBJECT (message),
				  "fullpath", file, g_free);

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::get_message: "
			      "message %p created from %s\n", message, file);
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::get_message\n");
	return message;
}

/**
 * CamelMaildirFolder::get_message_count: count messages in maildir
 * @folder:  the folder object
 *
 * Returns the number of messages in the maildir folder. New messages
 * are included in this count. 
 *
 * Return value: number of messages in the maildir, -1 on error
 */
static gint
_get_message_count (CamelFolder *folder, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER(folder);
	const gchar *maildir;
	gchar *newdir, *curdir, *newfile, *curfile;
	DIR *dir_handle;
	struct dirent *dir_entry;
	guint count = 0;

	CAMEL_LOG_FULL_DEBUG ("Entering "
			      "CamelMaildirFolder::get_message_count\n");
	g_assert(folder);

	/* check if the maildir exists */
	if (!camel_folder_exists (folder, ex)) return -1;

	maildir = maildir_folder->directory_path;

	newdir = g_strconcat (maildir, G_DIR_SEPARATOR_S, "new", NULL);
	curdir = g_strconcat (maildir, G_DIR_SEPARATOR_S, "cur", NULL);

	/* Check new messages */
	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::get_message_count: "
			      "getting new messages from %s\n", newdir);
	if ((dir_handle = _xopendir (newdir))) {
		while ((dir_entry = readdir (dir_handle))) {
			if (dir_entry->d_name[0] == '.') continue;
			newfile = g_strconcat (newdir, G_DIR_SEPARATOR_S,
					       dir_entry->d_name, NULL);
			curfile = g_strconcat (curdir, G_DIR_SEPARATOR_S,
					       dir_entry->d_name, ":2,", NULL);
			
			_xrename (newfile, curfile);
			
			g_free (curfile);
			g_free (newfile);
		}
		closedir (dir_handle);
	}

	/* Count messages */
	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::get_message_count: "
			      "counting messages in %s\n", curdir);
	if ((dir_handle = _xopendir (curdir))) {
		while ((dir_entry = readdir (dir_handle)))
			if (dir_entry->d_name[0] != '.') count++;
		closedir (dir_handle);
	}

	g_free (curdir);
	g_free (newdir);

	CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::get_message_count: "
			      " found %d messages\n", count);
	CAMEL_LOG_FULL_DEBUG ("Leaving "
			      "CamelMaildirFolder::get_message_count\n");
	return count;
}




/**
 * CamelMaildirFolder::expunge: expunge messages marked as deleted
 * @folder:  the folder object
 *
 * Physically deletes the messages marked as deleted in the folder.
 */
static void
_expunge (CamelFolder *folder, CamelException *ex)
{
	CamelMimeMessage *message;
	GList *node;
	gchar *fullpath;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::expunge\n");
	g_assert(folder);

	/* expunge messages marked for deletion */
	for (node = folder->message_list; node; node = g_list_next(node)) {
		message = CAMEL_MIME_MESSAGE (node->data);
		if (!message) {
			CAMEL_LOG_WARNING ("CamelMaildirFolder::expunge: "
					   "null message in node %p\n", node);
			continue;
		}
				
		if (camel_mime_message_get_flag (message, "DELETED")) {
			CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::expunge: "
					      "expunging message #%d\n",
					      message->message_number);

			/* expunge the message */
			fullpath = gtk_object_get_data (GTK_OBJECT (message),
							"fullpath");
			CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::expunge: "
					      "message fullpath is %s\n",
					      fullpath);

			if (_xunlink (fullpath))
				message->expunged = TRUE;
		} else {
			CAMEL_LOG_FULL_DEBUG ("CamelMaildirFolder::expunge: "
					      "skipping message #%d\n",
					      message->message_number);
		}
	}
	
	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::expunge\n");
}




/**
 * CamelMaildirFolder::list_subfolders: return a list of subfolders
 * @folder:  the folder object
 *
 * Returns the names of the maildir subfolders in a list.
 *
 * Return value: list of subfolder names
 */
static GList *
_list_subfolders (CamelFolder *folder, CamelException *ex)
{
	CamelMaildirFolder *maildir_folder = CAMEL_MAILDIR_FOLDER (folder);
	const gchar *maildir;
	gchar *subdir;
	struct stat statbuf;
	struct dirent *dir_entry;
	DIR *dir_handle;
	GList *subfolders = NULL;

	CAMEL_LOG_FULL_DEBUG ("Entering CamelMaildirFolder::list_subfolders\n");
	g_assert (folder);

	/* check if the maildir exists */
	if (!camel_folder_exists (folder, ex)) return NULL;

	/* scan through the maildir toplevel directory */
	maildir = maildir_folder->directory_path;
	if ((dir_handle = _xopendir (maildir))) {
		while ((dir_entry = readdir (dir_handle))) {
			if (dir_entry->d_name[0] == '.') continue;
			if (strcmp (dir_entry->d_name, "new") == 0) continue;
			if (strcmp (dir_entry->d_name, "cur") == 0) continue;
			if (strcmp (dir_entry->d_name, "tmp") == 0) continue;

			subdir = g_strconcat (maildir, G_DIR_SEPARATOR_S,
					      dir_entry->d_name, NULL);
			
			if (_xstat (subdir, &statbuf)
			    && S_ISDIR (statbuf.st_mode))
				subfolders =
					g_list_append (
						subfolders,
						g_strdup (dir_entry->d_name));
			
			g_free (subdir);
		}
		closedir (dir_handle);
	}

	CAMEL_LOG_FULL_DEBUG ("Leaving CamelMaildirFolder::list_subfolders\n");
	return subfolders;
}







/*
 * fs utility function 
 *
 */

static DIR *
_xopendir (const gchar *path)
{
	DIR *handle;
	g_assert (path);

	handle = opendir (path);
	if (!handle) {
		CAMEL_LOG_WARNING ("ERROR: opendir (%s);\n", path);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
	}

	return handle;
}

static gboolean
_xstat (const gchar *path, struct stat *buf)
{
	gint stat_error;
	g_assert (path);
	g_assert (buf);

	stat_error = stat (path, buf);
	if (stat_error == 0) {
		return TRUE;
	} else if (errno == ENOENT) {
		buf->st_mode = 0;
		return TRUE;
	} else {
		CAMEL_LOG_WARNING ("ERROR: stat (%s, %p);\n", path, buf);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
		return FALSE;
	}
}

static gboolean
_xmkdir (const gchar *path)
{
	g_assert (path);

	if (mkdir (path, S_IRWXU) == -1) {
		CAMEL_LOG_WARNING ("ERROR: mkdir (%s, S_IRWXU);\n", path);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
		return FALSE;
	} 

	return TRUE;
}

static gboolean
_xrename (const gchar *from, const gchar *to)
{
	g_assert (from);
	g_assert (to);

	if (rename (from, to) == 0) {
		return TRUE;
	} else {
		CAMEL_LOG_WARNING ("ERROR: rename (%s, %s);\n", from, to);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
		return FALSE;
	}
}

static gboolean
_xunlink (const gchar *path)
{
	g_assert (path);

	if (unlink (path) == 0) {
		return TRUE;
	} else if (errno == ENOENT) {
		return TRUE;
	} else {
		CAMEL_LOG_WARNING ("ERROR: unlink (%s);\n", path);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
		return FALSE;
	}
}

static gboolean
_xrmdir (const gchar *path)
{
	DIR *dir_handle;
	struct dirent *dir_entry;
	gchar *file;
	struct stat statbuf;
	g_assert (path);

	dir_handle = opendir (path);
	if (!dir_handle && errno == ENOENT) {
		return TRUE;
	} else if (!dir_handle) {
		CAMEL_LOG_WARNING ("ERROR: opendir (%s);\n", path);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
		return FALSE;
	}

	while ((dir_entry = readdir (dir_handle))) {
		file = g_strconcat (path, G_DIR_SEPARATOR_S, dir_entry->d_name,
				    NULL);
		if (_xstat (file, &statbuf) && S_ISREG (statbuf.st_mode))
			_xunlink (file);
		g_free (file);
	}

	closedir (dir_handle);

	if (rmdir (path) == 0) {
		return TRUE;
	} else if (errno == ENOENT) {
		return TRUE;
	} else {
		CAMEL_LOG_WARNING ("ERROR: rmdir (%s);\n", path);
		CAMEL_LOG_FULL_DEBUG ("  Full error text is: (%d) %s\n",
				      errno, strerror(errno));
		return FALSE;
	} 
}

/** *** **/