/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* e-path.c
*
* Copyright (C) 2001 Ximian, Inc.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of version 2 of the GNU General Public
* License as published by the Free Software Foundation.
*
* 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.
*/
#include <config.h>
#include <sys/types.h>
#include <dirent.h>
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>
#include <glib.h>
#include "e-path.h"
#define SUBFOLDER_DIR_NAME "subfolders"
#define SUBFOLDER_DIR_NAME_LEN 10
/**
* e_path_to_physical:
* @prefix: a prefix to prepend to the path, or %NULL
* @path: the virtual path to convert to a filesystem path.
*
* This converts the "virtual" path @path into an expanded form that
* allows a given name to refer to both a file and a directory. The
* expanded path will have a "subfolders" directory inserted between
* each path component. If the path ends with "/", the returned
* physical path will end with "/subfolders"
*
* If @prefix is non-%NULL, it will be prepended to the returned path.
*
* Return value: the expanded path
**/
char *
e_path_to_physical (const char *prefix, const char *vpath)
{
const char *p, *newp;
char *dp;
char *ppath;
int ppath_len;
int prefix_len;
while (*vpath == '/')
vpath++;
if (!prefix)
prefix = "";
/* Calculate the length of the real path. */
ppath_len = strlen (vpath);
ppath_len++; /* For the ending zero. */
prefix_len = strlen (prefix);
ppath_len += prefix_len;
ppath_len++; /* For the separating slash. */
/* Take account of the fact that we need to translate every
* separator into `subfolders/'.
*/
p = vpath;
while (1) {
newp = strchr (p, '/');
if (newp == NULL)
break;
ppath_len += SUBFOLDER_DIR_NAME_LEN;
ppath_len++; /* For the separating slash. */
/* Skip consecutive slashes. */
while (*newp == '/')
newp++;
p = newp;
};
ppath = g_malloc (ppath_len);
dp = ppath;
memcpy (dp, prefix, prefix_len);
dp += prefix_len;
*(dp++) = '/';
/* Copy the mangled path. */
p = vpath;
while (1) {
newp = strchr (p, '/');
if (newp == NULL) {
strcpy (dp, p);
break;
}
memcpy (dp, p, newp - p + 1); /* `+ 1' to copy the slash too. */
dp += newp - p + 1;
memcpy (dp, SUBFOLDER_DIR_NAME, SUBFOLDER_DIR_NAME_LEN);
dp += SUBFOLDER_DIR_NAME_LEN;
*(dp++) = '/';
/* Skip consecutive slashes. */
while (*newp == '/')
newp++;
p = newp;
}
return ppath;
}
static gboolean
find_folders_recursive (const char *physical_path, const char *path,
EPathFindFoldersCallback callback, gpointer data)
{
DIR *dir;
char *subfolder_directory_path;
gboolean ok;
if (*path) {
if (!callback (physical_path, path, data))
return FALSE;
subfolder_directory_path = g_strdup_printf ("%s/%s", physical_path, SUBFOLDER_DIR_NAME);
} else {
/* On the top level, we have no folders and,
* consequently, no subfolder directory.
*/
subfolder_directory_path = g_strdup (physical_path);
}
/* Now scan the subfolders and load them. */
dir = opendir (subfolder_directory_path);
if (dir == NULL) {
g_free (subfolder_directory_path);
return TRUE;
}
ok = TRUE;
while (ok) {
struct stat file_stat;
struct dirent *dirent;
char *file_path;
char *new_path;
dirent = readdir (dir);
if (dirent == NULL)
break;
if (strcmp (dirent->d_name, ".") == 0 || strcmp (dirent->d_name, "..") == 0)
continue;
file_path = g_strdup_printf ("%s/%s", subfolder_directory_path,
dirent->d_name);
if (stat (file_path, &file_stat) < 0 ||
! S_ISDIR (file_stat.st_mode)) {
g_free (file_path);
continue;
}
new_path = g_strdup_printf ("%s/%s", path, dirent->d_name);
ok = find_folders_recursive (file_path, new_path, callback, data);
g_free (file_path);
g_free (new_path);
}
closedir (dir);
g_free (subfolder_directory_path);
return ok;
}
/**
* e_path_find_folders:
* @prefix: directory to start from
* @callback: Callback to invoke on each folder
* @data: Data for @callback
*
* Walks the folder tree starting at @prefix and calls @callback
* on each folder.
*
* Return value: %TRUE on success, %FALSE if an error occurs at any point
**/
gboolean
e_path_find_folders (const char *prefix,
EPathFindFoldersCallback callback,
gpointer data)
{
return find_folders_recursive (prefix, "", callback, data);
}
/**
* e_path_rmdir:
* @prefix: a prefix to prepend to the path, or %NULL
* @path: the virtual path to convert to a filesystem path.
*
* This removes the directory pointed to by @prefix and @path
* and attempts to remove its parent "subfolders" directory too
* if it's empty.
*
* Return value: -1 (with errno set) if it failed to rmdir the
* specified directory. 0 otherwise, whether or not it removed
* the parent directory.
**/
int
e_path_rmdir (const char *prefix, const char *vpath)
{
char *physical_path, *p;
/* Remove the directory itself */
physical_path = e_path_to_physical (prefix, vpath);
if (rmdir (physical_path) == -1) {
g_free (physical_path);
return -1;
}
/* Attempt to remove its parent "subfolders" directory,
* ignoring errors since it might not be empty.
*/
p = strrchr (physical_path, '/');
if (p[1] == '\0') {
g_free (physical_path);
return 0;
}
*p = '\0';
p = strrchr (physical_path, '/');
if (!p || strcmp (p + 1, SUBFOLDER_DIR_NAME) != 0) {
g_free (physical_path);
return 0;
}
rmdir (physical_path);
g_free (physical_path);
return 0;
}