Blame gegl/module/gegldatafiles.c

Packit Service 2781ba
/* this file is part of GEGL
Packit Service 2781ba
 *
Packit Service 2781ba
 * Datafiles module copyight (C) 1996 Federico Mena Quintero
Packit Service 2781ba
 * federico@nuclecu.unam.mx
Packit Service 2781ba
 *
Packit Service 2781ba
 * This library is free software; you can redistribute it and/or
Packit Service 2781ba
 * modify it under the terms of the GNU Lesser General Public
Packit Service 2781ba
 * License as published by the Free Software Foundation; either
Packit Service 2781ba
 * version 3 of the License, or (at your option) any later version.
Packit Service 2781ba
 *
Packit Service 2781ba
 * This library is distributed in the hope that it will be useful,
Packit Service 2781ba
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 2781ba
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 2781ba
 * Lesser General Public License for more details.
Packit Service 2781ba
 *
Packit Service 2781ba
 * You should have received a copy of the GNU Lesser General Public
Packit Service 2781ba
 * License along with GEGL; if not, see <http://www.gnu.org/licenses/>.
Packit Service 2781ba
 *
Packit Service 2781ba
 * Datafiles module copyight (C) 1996 Federico Mena Quintero
Packit Service 2781ba
 * federico@nuclecu.unam.mx
Packit Service 2781ba
 */
Packit Service 2781ba
Packit Service 2781ba
#include "config.h"
Packit Service 2781ba
Packit Service 2781ba
#include <string.h>
Packit Service 2781ba
Packit Service 2781ba
#include <sys/types.h>
Packit Service 2781ba
#ifdef HAVE_UNISTD_H
Packit Service 2781ba
#include <unistd.h>
Packit Service 2781ba
#endif
Packit Service 2781ba
Packit Service 2781ba
#include <glib-object.h>
Packit Service 2781ba
#include <glib/gstdio.h>
Packit Service 2781ba
Packit Service 2781ba
#ifdef G_OS_WIN32
Packit Service 2781ba
#ifndef S_ISREG
Packit Service 2781ba
#define S_ISREG(m) (((m) & _S_IFMT) == _S_IFREG)
Packit Service 2781ba
#endif
Packit Service 2781ba
#ifndef S_ISDIR
Packit Service 2781ba
#define S_ISDIR(m) (((m) & _S_IFMT) == _S_IFDIR)
Packit Service 2781ba
#endif
Packit Service 2781ba
#ifndef S_IXUSR
Packit Service 2781ba
#define S_IXUSR _S_IEXEC
Packit Service 2781ba
#endif
Packit Service 2781ba
#endif
Packit Service 2781ba
Packit Service 2781ba
/*
Packit Service 2781ba
#include "geglbasetypes.h"*/
Packit Service 2781ba
Packit Service 2781ba
#include "gegldatafiles.h"
Packit Service 2781ba
Packit Service 2781ba
gboolean
Packit Service 2781ba
gegl_datafiles_check_extension (const gchar *filename,
Packit Service 2781ba
                                const gchar *extension)
Packit Service 2781ba
{
Packit Service 2781ba
  gint name_len;
Packit Service 2781ba
  gint ext_len;
Packit Service 2781ba
Packit Service 2781ba
  g_return_val_if_fail (filename != NULL, FALSE);
Packit Service 2781ba
  g_return_val_if_fail (extension != NULL, FALSE);
Packit Service 2781ba
Packit Service 2781ba
  name_len = strlen (filename);
Packit Service 2781ba
  ext_len  = strlen (extension);
Packit Service 2781ba
Packit Service 2781ba
  if (! (name_len && ext_len && (name_len > ext_len)))
Packit Service 2781ba
    return FALSE;
Packit Service 2781ba
Packit Service 2781ba
  return (g_ascii_strcasecmp (&filename[name_len - ext_len], extension) == 0);
Packit Service 2781ba
}
Packit Service 2781ba
Packit Service 2781ba
/**
Packit Service 2781ba
 * gegl_path_parse:
Packit Service 2781ba
 * @path:         A list of directories separated by #G_SEARCHPATH_SEPARATOR.
Packit Service 2781ba
 * @max_paths:    The maximum number of directories to return.
Packit Service 2781ba
 * @check:        %TRUE if you want the directories to be checked.
Packit Service 2781ba
 * @check_failed: Returns a #GList of path elements for which the
Packit Service 2781ba
 *                check failed.
Packit Service 2781ba
 *
Packit Service 2781ba
 * Returns: A #GList of all directories in @path.
Packit Service 2781ba
 **/
Packit Service 2781ba
static GList *
Packit Service 2781ba
gegl_path_parse (const gchar  *path,
Packit Service 2781ba
                 gint          max_paths,
Packit Service 2781ba
                 gboolean      check,
Packit Service 2781ba
                 GList       **check_failed)
Packit Service 2781ba
{
Packit Service 2781ba
  const gchar  *home;
Packit Service 2781ba
  gchar       **patharray;
Packit Service 2781ba
  GList        *list      = NULL;
Packit Service 2781ba
  GList        *fail_list = NULL;
Packit Service 2781ba
  gint          i;
Packit Service 2781ba
  gboolean      exists    = TRUE;
Packit Service 2781ba
Packit Service 2781ba
  if (!path || !*path || max_paths < 1 || max_paths > 256)
Packit Service 2781ba
    return NULL;
Packit Service 2781ba
Packit Service 2781ba
  home = g_get_home_dir ();
Packit Service 2781ba
Packit Service 2781ba
  patharray = g_strsplit (path, G_SEARCHPATH_SEPARATOR_S, max_paths);
Packit Service 2781ba
Packit Service 2781ba
  for (i = 0; i < max_paths; i++)
Packit Service 2781ba
    {
Packit Service 2781ba
      GString *dir;
Packit Service 2781ba
Packit Service 2781ba
      if (! patharray[i])
Packit Service 2781ba
        break;
Packit Service 2781ba
Packit Service 2781ba
#ifndef G_OS_WIN32
Packit Service 2781ba
      if (*patharray[i] == '~')
Packit Service 2781ba
        {
Packit Service 2781ba
          dir = g_string_new (home);
Packit Service 2781ba
          g_string_append (dir, patharray[i] + 1);
Packit Service 2781ba
        }
Packit Service 2781ba
      else
Packit Service 2781ba
#endif
Packit Service 2781ba
        {
Packit Service 2781ba
          dir = g_string_new (patharray[i]);
Packit Service 2781ba
        }
Packit Service 2781ba
Packit Service 2781ba
      if (check)
Packit Service 2781ba
        exists = g_file_test (dir->str, G_FILE_TEST_IS_DIR);
Packit Service 2781ba
Packit Service 2781ba
      if (exists)
Packit Service 2781ba
        list = g_list_prepend (list, g_strdup (dir->str));
Packit Service 2781ba
      else if (check_failed)
Packit Service 2781ba
        fail_list = g_list_prepend (fail_list, g_strdup (dir->str));
Packit Service 2781ba
Packit Service 2781ba
      g_string_free (dir, TRUE);
Packit Service 2781ba
    }
Packit Service 2781ba
Packit Service 2781ba
  g_strfreev (patharray);
Packit Service 2781ba
Packit Service 2781ba
  list = g_list_reverse (list);
Packit Service 2781ba
Packit Service 2781ba
  if (check && check_failed)
Packit Service 2781ba
    {
Packit Service 2781ba
      fail_list = g_list_reverse (fail_list);
Packit Service 2781ba
      *check_failed = fail_list;
Packit Service 2781ba
    }
Packit Service 2781ba
Packit Service 2781ba
  return list;
Packit Service 2781ba
}
Packit Service 2781ba
Packit Service 2781ba
/**
Packit Service 2781ba
 * gegl_path_free:
Packit Service 2781ba
 * @path: A list of directories as returned by gegl_path_parse().
Packit Service 2781ba
 *
Packit Service 2781ba
 * This function frees the memory allocated for the list and the strings
Packit Service 2781ba
 * it contains.
Packit Service 2781ba
 **/
Packit Service 2781ba
static void
Packit Service 2781ba
gegl_path_free (GList *path)
Packit Service 2781ba
{
Packit Service 2781ba
  g_list_foreach (path, (GFunc) g_free, NULL);
Packit Service 2781ba
  g_list_free (path);
Packit Service 2781ba
}
Packit Service 2781ba
Packit Service 2781ba
void
Packit Service 2781ba
gegl_datafiles_read_directories (const gchar            *path_str,
Packit Service 2781ba
                                 GFileTest               flags,
Packit Service 2781ba
                                 GeglDatafileLoaderFunc  loader_func,
Packit Service 2781ba
                                 gpointer                user_data)
Packit Service 2781ba
{
Packit Service 2781ba
  GeglDatafileData  file_data;
Packit Service 2781ba
  struct stat       filestat;
Packit Service 2781ba
  gchar            *local_path;
Packit Service 2781ba
  GList            *path;
Packit Service 2781ba
  GList            *list;
Packit Service 2781ba
  gchar            *filename;
Packit Service 2781ba
  gint              err;
Packit Service 2781ba
  GDir             *dir;
Packit Service 2781ba
  const gchar      *dir_ent;
Packit Service 2781ba
Packit Service 2781ba
  g_return_if_fail (path_str != NULL);
Packit Service 2781ba
  g_return_if_fail (loader_func != NULL);
Packit Service 2781ba
Packit Service 2781ba
  local_path = g_strdup (path_str);
Packit Service 2781ba
Packit Service 2781ba
  path = gegl_path_parse (local_path, 16, TRUE, NULL);
Packit Service 2781ba
Packit Service 2781ba
  for (list = path; list; list = g_list_next (list))
Packit Service 2781ba
    {
Packit Service 2781ba
      const gchar *dirname = list->data;
Packit Service 2781ba
Packit Service 2781ba
      dir = g_dir_open (dirname, 0, NULL);
Packit Service 2781ba
Packit Service 2781ba
      if (dir)
Packit Service 2781ba
        {
Packit Service 2781ba
          while ((dir_ent = g_dir_read_name (dir)))
Packit Service 2781ba
            {
Packit Service 2781ba
              filename = g_build_filename (dirname, dir_ent, NULL);
Packit Service 2781ba
Packit Service 2781ba
              err = g_stat (filename, &filestat);
Packit Service 2781ba
Packit Service 2781ba
              file_data.filename = filename;
Packit Service 2781ba
              file_data.dirname  = dirname;
Packit Service 2781ba
              file_data.basename = dir_ent;
Packit Service 2781ba
              file_data.atime    = filestat.st_atime;
Packit Service 2781ba
              file_data.mtime    = filestat.st_mtime;
Packit Service 2781ba
              file_data.ctime    = filestat.st_ctime;
Packit Service 2781ba
Packit Service 2781ba
              if (! err)
Packit Service 2781ba
                {
Packit Service 2781ba
                  if (/*(flags & G_FILE_TEST_IS_DIR) &&*/
Packit Service 2781ba
                           S_ISDIR (filestat.st_mode))
Packit Service 2781ba
                    {
Packit Service 2781ba
                      gegl_datafiles_read_directories (filename,
Packit Service 2781ba
                                                       flags,
Packit Service 2781ba
                                                       loader_func,
Packit Service 2781ba
                                                       user_data);
Packit Service 2781ba
                    }
Packit Service 2781ba
                  else if (flags & G_FILE_TEST_EXISTS)
Packit Service 2781ba
                    {
Packit Service 2781ba
                      (* loader_func) (&file_data, user_data);
Packit Service 2781ba
                    }
Packit Service 2781ba
                  else if ((flags & G_FILE_TEST_IS_REGULAR) &&
Packit Service 2781ba
                           S_ISREG (filestat.st_mode))
Packit Service 2781ba
                    {
Packit Service 2781ba
                      (* loader_func) (&file_data, user_data);
Packit Service 2781ba
                    }
Packit Service 2781ba
#ifndef G_OS_WIN32
Packit Service 2781ba
                  else if ((flags & G_FILE_TEST_IS_SYMLINK) &&
Packit Service 2781ba
                           S_ISLNK (filestat.st_mode))
Packit Service 2781ba
                    {
Packit Service 2781ba
                      (* loader_func) (&file_data, user_data);
Packit Service 2781ba
                    }
Packit Service 2781ba
#endif
Packit Service 2781ba
                  else if ((flags & G_FILE_TEST_IS_EXECUTABLE) &&
Packit Service 2781ba
                           (((filestat.st_mode & S_IXUSR) &&
Packit Service 2781ba
                             !S_ISDIR (filestat.st_mode)) ||
Packit Service 2781ba
                            (S_ISREG (filestat.st_mode) /*&&
Packit Service 2781ba
                             is_script (filename)*/)))
Packit Service 2781ba
                    {
Packit Service 2781ba
                      (* loader_func) (&file_data, user_data);
Packit Service 2781ba
                    }
Packit Service 2781ba
                }
Packit Service 2781ba
Packit Service 2781ba
              g_free (filename);
Packit Service 2781ba
            }
Packit Service 2781ba
Packit Service 2781ba
          g_dir_close (dir);
Packit Service 2781ba
        }
Packit Service 2781ba
    }
Packit Service 2781ba
Packit Service 2781ba
  gegl_path_free (path);
Packit Service 2781ba
  g_free (local_path);
Packit Service 2781ba
}