efl/legacy/ecore/src/lib/ecore/ecore_path.c

207 lines
4.3 KiB
C

/*
* vim:ts=8:sw=3:sts=8:noexpandtab:cino=>5n-3f0^-2{2
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <dirent.h>
#include "ecore_private.h"
#include "Ecore_Data.h"
#include "Ecore_Str.h"
/**
* @defgroup Ecore_Path_Group Path Group Functions
*
* Functions that make it easier to find a file in a set of search paths.
*
* @todo Give this a better description.
*/
/**
* Creates a new path group.
* @param group_name The name of the new group.
* @return @c NULL on error, the handle of the new group on success.
* @ingroup Ecore_Path_Group
*/
EAPI Ecore_Path_Group *
ecore_path_group_new(void)
{
return calloc(1, sizeof(Ecore_Path_Group));
}
/**
* Destroys a previously created path group
* @param group The group to delete.
* @ingroup Ecore_Path_Group
*/
EAPI void
ecore_path_group_del(Ecore_Path_Group *group)
{
char *path;
CHECK_PARAM_POINTER("group", group);
EINA_LIST_FREE(group->paths, path)
free(path);
free(group);
}
/**
* Adds a directory to be searched for files.
* @param group The group to add the path.
* @param path The new path to be added to the group.
* @ingroup Ecore_Path_Group
*/
EAPI void
ecore_path_group_add(Ecore_Path_Group *group, const char *path)
{
CHECK_PARAM_POINTER("group", group);
CHECK_PARAM_POINTER("path", path);
group->paths = eina_list_append(group->paths, strdup(path));
}
/**
* Removes the given directory from the given group.
* @param group The group to remove the path from.
* @param path The path of the directory to be removed.
* @ingroup Ecore_Path_Group
*/
EAPI void
ecore_path_group_remove(Ecore_Path_Group *group, const char *path)
{
char *found;
CHECK_PARAM_POINTER("group", group);
CHECK_PARAM_POINTER("path", path);
if (!group->paths)
return;
/*
* Find the path in the list of available paths
*/
found = eina_list_search_unsorted(group->paths, EINA_COMPARE_CB(strcmp), path);
/*
* If the path is found, remove and free it
*/
if (found)
{
group->paths = eina_list_remove(group->paths, found);
free(found);
}
}
/**
* Finds a file in a group of paths.
* @param group The group to search.
* @param name The name of the file to find.
* @return A pointer to a newly allocated path location of the found file
* on success. @c NULL on failure.
* @ingroup Ecore_Path_Group
*/
EAPI char *
ecore_path_group_find(Ecore_Path_Group *group, const char *name)
{
Eina_List *l;
int r;
char *p;
struct stat st;
char path[PATH_MAX];
CHECK_PARAM_POINTER_RETURN("group", group, NULL);
CHECK_PARAM_POINTER_RETURN("name", name, NULL);
if (!group->paths)
return NULL;
/*
* Search the paths of the path group for the specified file name
*/
EINA_LIST_FOREACH(group->paths, l, p)
{
snprintf(path, PATH_MAX, "%s/%s", p, name);
r = stat(path, &st);
if ((r >= 0) && S_ISREG(st.st_mode))
break;
}
if (p)
p = strdup(path);
return p;
}
/**
* Retrieves a list of all available files in the given path.
* @param group_id The identifier for the given path.
* @return A pointer to a newly allocated list of all files found in the paths
* identified by @p group_id. @c NULL otherwise.
* @ingroup Ecore_Path_Group
*/
EAPI Eina_List *
ecore_path_group_available_get(Ecore_Path_Group *group)
{
Eina_List *avail = NULL;
Eina_List *l;
char *path;
CHECK_PARAM_POINTER_RETURN("group", group, NULL);
if (!group->paths || !eina_list_count(group->paths))
return NULL;
EINA_LIST_FOREACH(group->paths, l, path)
{
DIR *dir;
struct stat st;
struct dirent *d;
stat(path, &st);
if (!S_ISDIR(st.st_mode))
continue;
dir = opendir(path);
if (!dir)
continue;
while ((d = readdir(dir)) != NULL)
{
char ppath[PATH_MAX];
/* char n[PATH_MAX];
int l;
*/
if (!strncmp(d->d_name, ".", 1))
continue;
snprintf(ppath, PATH_MAX, "%s/%s", path, d->d_name);
stat(ppath, &st);
if (!S_ISREG(st.st_mode))
continue;
/*
l = strlen(d->d_name);
strncpy(n, d->d_name, l - 2);
*/
/* avail = eina_list_append(avail, strdup(n));*/
avail = eina_list_append(avail, strdup(d->d_name));
}
}
return avail;
}