207 lines
4.7 KiB
C
207 lines
4.7 KiB
C
/* -*- 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 <dirent.h>
|
|
#include <string.h>
|
|
#include <sys/stat.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);
|
|
}
|