1244 lines
30 KiB
C++
1244 lines
30 KiB
C++
/*
|
|
* mooutils-fs.c
|
|
*
|
|
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
|
|
*
|
|
* This file is part of medit. medit is free software; you can
|
|
* redistribute it and/or modify it under the terms of the
|
|
* GNU Lesser General Public License as published by the
|
|
* Free Software Foundation; either version 2.1 of the License,
|
|
* or (at your option) any later version.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include "mooutils/mooutils-fs.h"
|
|
#include "mooutils/mooutils-debug.h"
|
|
#include "mooutils/mooutils-mem.h"
|
|
#include "mooutils/mootype-macros.h"
|
|
#include "mooutils/mooi18n.h"
|
|
#include <mooutils/mooutils-tests.h>
|
|
#include <moocpp/moocpp.h>
|
|
#include <mooglib/moo-stat.h>
|
|
#include <mooglib/moo-glib.h>
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <sys/types.h>
|
|
|
|
#ifdef __WIN32__
|
|
#include <windows.h>
|
|
#include <io.h>
|
|
#include <direct.h>
|
|
#include <shellapi.h>
|
|
#include <gdk/gdkwin32.h>
|
|
#endif
|
|
|
|
#ifndef S_IRWXU
|
|
#define S_IRWXU 0
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_WAIT_H
|
|
#include <sys/wait.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
using namespace moo;
|
|
|
|
#define BROKEN_NAME "<" "????" ">"
|
|
|
|
MOO_DEFINE_QUARK (moo-file-error-quark, _moo_file_error_quark)
|
|
|
|
/* XXX fix this */
|
|
gboolean
|
|
_moo_save_file_utf8 (const char *name,
|
|
const char *text,
|
|
gssize len,
|
|
GError **error)
|
|
{
|
|
GIOChannel *file;
|
|
GIOStatus status;
|
|
gsize bytes_written;
|
|
gsize real_len;
|
|
|
|
file = g_io_channel_new_file (name, "w", error);
|
|
|
|
if (!file)
|
|
return FALSE;
|
|
|
|
real_len = len < 0 ? strlen (text) : (gsize) len;
|
|
|
|
status = g_io_channel_write_chars (file, text,
|
|
len, &bytes_written,
|
|
error);
|
|
|
|
if (status != G_IO_STATUS_NORMAL || bytes_written != real_len)
|
|
{
|
|
/* glib #320668 */
|
|
g_io_channel_flush (file, NULL);
|
|
g_io_channel_shutdown (file, FALSE, NULL);
|
|
g_io_channel_unref (file);
|
|
return FALSE;
|
|
}
|
|
|
|
/* glib #320668 */
|
|
g_io_channel_flush (file, NULL);
|
|
g_io_channel_shutdown (file, FALSE, NULL);
|
|
g_io_channel_unref (file);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
#ifdef __WIN32__
|
|
|
|
static wchar_t *
|
|
filename_to_double_null_terminated_string (const char *filename)
|
|
{
|
|
long len;
|
|
gunichar2 *ret;
|
|
|
|
ret = g_utf8_to_utf16 (filename, -1, NULL, &len, NULL);
|
|
|
|
if (!ret)
|
|
return NULL;
|
|
|
|
ret = g_renew (gunichar2, ret, len + 2);
|
|
ret[len + 1] = 0;
|
|
|
|
return (wchar_t*) ret;
|
|
}
|
|
|
|
static wchar_t *
|
|
filename_list_to_double_null_terminated_string (GList *filenames)
|
|
{
|
|
GArray *ret;
|
|
gunichar2 wzero = 0;
|
|
|
|
ret = g_array_new (FALSE, FALSE, sizeof (gunichar2));
|
|
|
|
while (filenames)
|
|
{
|
|
guint old_len;
|
|
long len;
|
|
gunichar2 *wstr;
|
|
|
|
wstr = g_utf8_to_utf16 (reinterpret_cast<char*> (filenames->data), -1, NULL, &len, NULL);
|
|
filenames = filenames->next;
|
|
|
|
if (!wstr)
|
|
{
|
|
g_array_free (ret, TRUE);
|
|
return NULL;
|
|
}
|
|
|
|
old_len = ret->len;
|
|
g_array_set_size (ret, old_len + len + 1);
|
|
memcpy (ret->data + old_len * sizeof (gunichar2), wstr, (len + 1) * sizeof (gunichar2));
|
|
|
|
g_free (wstr);
|
|
}
|
|
|
|
g_array_append_val (ret, wzero);
|
|
return (wchar_t*) g_array_free (ret, FALSE);
|
|
}
|
|
|
|
static gboolean
|
|
move_or_copy_files_ui (GList *filenames,
|
|
const char *destdir,
|
|
gboolean copy,
|
|
GtkWidget *parent)
|
|
{
|
|
SHFILEOPSTRUCT shop = {0};
|
|
gboolean ret;
|
|
wchar_t *from, *to;
|
|
|
|
from = filename_list_to_double_null_terminated_string (filenames);
|
|
to = filename_to_double_null_terminated_string (destdir);
|
|
|
|
if (!from || !to)
|
|
{
|
|
g_free (from);
|
|
g_free (to);
|
|
return FALSE;
|
|
}
|
|
|
|
shop.hwnd = parent && parent->window ? ((HWND) GDK_WINDOW_HWND (parent->window)) : nullptr;
|
|
shop.wFunc = copy ? FO_COPY : FO_MOVE;
|
|
shop.pFrom = from;
|
|
shop.pTo = to;
|
|
shop.fFlags = FOF_ALLOWUNDO;
|
|
|
|
ret = SHFileOperation(&shop) == 0 && !shop.fAnyOperationsAborted;
|
|
|
|
g_free (from);
|
|
g_free (to);
|
|
return ret;
|
|
}
|
|
|
|
gboolean
|
|
_moo_copy_files_ui (GList *filenames,
|
|
const char *destdir,
|
|
GtkWidget *parent)
|
|
{
|
|
return move_or_copy_files_ui (filenames, destdir, TRUE, parent);
|
|
}
|
|
|
|
gboolean
|
|
_moo_move_files_ui (GList *filenames,
|
|
const char *destdir,
|
|
GtkWidget *parent)
|
|
{
|
|
return move_or_copy_files_ui (filenames, destdir, FALSE, parent);
|
|
}
|
|
|
|
#endif /* __WIN32__ */
|
|
|
|
|
|
#ifndef __WIN32__
|
|
static gboolean
|
|
rm_fr (const char *path,
|
|
GError **error)
|
|
{
|
|
GError *error_here = NULL;
|
|
char **argv;
|
|
char *child_err;
|
|
int status;
|
|
|
|
argv = g_new0 (char*, 5);
|
|
argv[0] = g_strdup ("/bin/rm");
|
|
argv[1] = g_strdup ("-fr");
|
|
argv[2] = g_strdup ("--");
|
|
argv[3] = g_strdup (path);
|
|
|
|
if (!g_spawn_sync (NULL, argv, NULL, G_SPAWN_STDOUT_TO_DEV_NULL,
|
|
NULL, NULL, NULL, &child_err, &status, &error_here))
|
|
{
|
|
g_set_error (error, MOO_FILE_ERROR, MOO_FILE_ERROR_FAILED,
|
|
/* This is error message in file selector when rm
|
|
fails for some strange reason */
|
|
_("Could not run 'rm' command: %s"),
|
|
error_here->message);
|
|
g_error_free (error_here);
|
|
g_strfreev (argv);
|
|
return FALSE;
|
|
}
|
|
|
|
g_strfreev (argv);
|
|
|
|
if (!WIFEXITED (status) || WEXITSTATUS (status))
|
|
{
|
|
if (child_err && strlen (child_err) > 5000)
|
|
strcpy (child_err + 4997, "...");
|
|
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_FAILED,
|
|
/* This is error message in file selector when rm
|
|
fails to delete a file or folder */
|
|
_("'rm' command failed: %s"),
|
|
MOO_NZS (child_err));
|
|
|
|
g_free (child_err);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
g_free (child_err);
|
|
return TRUE;
|
|
}
|
|
}
|
|
#endif /* __WIN32__ */
|
|
|
|
|
|
#ifdef __WIN32__
|
|
static gboolean
|
|
rm_r (const char *path,
|
|
GError **error)
|
|
{
|
|
GDir *dir;
|
|
const char *file;
|
|
gboolean success = TRUE;
|
|
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
|
|
dir = g_dir_open (path, 0, error);
|
|
|
|
if (!dir)
|
|
return FALSE;
|
|
|
|
while (success && (file = g_dir_read_name (dir)))
|
|
{
|
|
char *file_path = g_build_filename (path, file, NULL);
|
|
|
|
if (g_file_test (file_path, G_FILE_TEST_IS_DIR))
|
|
{
|
|
if (!rm_r (file_path, error))
|
|
success = FALSE;
|
|
}
|
|
else
|
|
{
|
|
mgw_errno_t err;
|
|
|
|
if (mgw_remove (file_path, &err) != 0)
|
|
{
|
|
success = FALSE;
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
_moo_file_error_from_errno (err),
|
|
_("Could not remove %s: %s"), file_path,
|
|
mgw_strerror (err));
|
|
}
|
|
}
|
|
|
|
g_free (file_path);
|
|
}
|
|
|
|
g_dir_close (dir);
|
|
|
|
if (success)
|
|
{
|
|
mgw_errno_t err;
|
|
|
|
if (mgw_remove (path, &err) != 0)
|
|
{
|
|
success = FALSE;
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
_moo_file_error_from_errno (err),
|
|
_("Could not remove %s: %s"), path,
|
|
mgw_strerror (err));
|
|
}
|
|
}
|
|
|
|
return success;
|
|
}
|
|
#endif
|
|
|
|
|
|
gboolean
|
|
_moo_remove_dir (const char *path,
|
|
gboolean recursive,
|
|
GError **error)
|
|
{
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
|
|
if (!recursive)
|
|
{
|
|
mgw_errno_t err;
|
|
|
|
if (mgw_remove (path, &err) != 0)
|
|
{
|
|
char *path_utf8 = g_filename_display_name (path);
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
_moo_file_error_from_errno (err),
|
|
_("Could not remove %s: %s"),
|
|
path_utf8, mgw_strerror (err));
|
|
g_free (path_utf8);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
#ifndef __WIN32__
|
|
return rm_fr (path, error);
|
|
#else
|
|
return rm_r (path, error);
|
|
#endif
|
|
}
|
|
|
|
|
|
int
|
|
_moo_mkdir_with_parents (const char *path, mgw_errno_t* err)
|
|
{
|
|
return mgw_mkdir_with_parents (path, S_IRWXU, err);
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_create_dir (const char *path,
|
|
GError **error)
|
|
{
|
|
MgwStatBuf buf;
|
|
char *utf8_path;
|
|
mgw_errno_t err;
|
|
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
|
|
if (mgw_stat (path, &buf, &err) != 0 && err.value != MGW_ENOENT)
|
|
{
|
|
utf8_path = g_filename_display_name (path);
|
|
|
|
g_set_error (error,
|
|
MOO_FILE_ERROR,
|
|
_moo_file_error_from_errno (err),
|
|
_("Could not create folder %s: %s"),
|
|
utf8_path, mgw_strerror (err));
|
|
|
|
g_free (utf8_path);
|
|
return FALSE;
|
|
}
|
|
|
|
if (mgw_errno_is_set (err))
|
|
{
|
|
if (_moo_mkdir (path, &err) == -1)
|
|
{
|
|
utf8_path = g_filename_display_name (path);
|
|
|
|
g_set_error (error,
|
|
MOO_FILE_ERROR,
|
|
_moo_file_error_from_errno (err),
|
|
_("Could not create folder %s: %s"),
|
|
utf8_path, mgw_strerror (err));
|
|
|
|
g_free (utf8_path);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
if (buf.isdir)
|
|
return TRUE;
|
|
|
|
utf8_path = g_filename_display_name (path);
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_ALREADY_EXISTS,
|
|
_("Could not create folder %s: %s"),
|
|
utf8_path, mgw_strerror (MGW_E_EXIST));
|
|
g_free (utf8_path);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_rename_file (const char *path,
|
|
const char *new_path,
|
|
GError **error)
|
|
{
|
|
// Do not break this for directories!
|
|
mgw_errno_t err;
|
|
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
g_return_val_if_fail (new_path != NULL, FALSE);
|
|
|
|
if (mgw_rename (path, new_path, &err) != 0)
|
|
{
|
|
char *utf8_path = g_filename_display_name (path);
|
|
char *utf8_new_path = g_filename_display_name (new_path);
|
|
|
|
g_set_error (error,
|
|
MOO_FILE_ERROR,
|
|
_moo_file_error_from_errno (err),
|
|
_("Could not rename file %s to %s: %s"),
|
|
utf8_path, utf8_new_path, mgw_strerror (err));
|
|
|
|
g_free (utf8_path);
|
|
g_free (utf8_new_path);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
MooFileError
|
|
_moo_file_error_from_errno (mgw_errno_t code)
|
|
{
|
|
switch (code.value)
|
|
{
|
|
case MGW_EACCES:
|
|
case MGW_EPERM:
|
|
return MOO_FILE_ERROR_ACCESS_DENIED;
|
|
case MGW_EEXIST:
|
|
return MOO_FILE_ERROR_ALREADY_EXISTS;
|
|
#ifndef __WIN32__
|
|
case MGW_ELOOP:
|
|
#endif
|
|
case MGW_ENAMETOOLONG:
|
|
return MOO_FILE_ERROR_BAD_FILENAME;
|
|
case MGW_ENOENT:
|
|
return MOO_FILE_ERROR_NONEXISTENT;
|
|
case MGW_ENOTDIR:
|
|
return MOO_FILE_ERROR_NOT_FOLDER;
|
|
case MGW_EROFS:
|
|
return MOO_FILE_ERROR_READONLY;
|
|
case MGW_EXDEV:
|
|
return MOO_FILE_ERROR_DIFFERENT_FS;
|
|
|
|
default:
|
|
return MOO_FILE_ERROR_FAILED;
|
|
}
|
|
}
|
|
|
|
|
|
static gstr
|
|
moo_filename_from_locale (const gstr& file)
|
|
{
|
|
g_return_val_if_fail (!file.empty (), gstr::null);
|
|
#ifdef __WIN32__
|
|
return g::locale_to_utf8 (file);
|
|
#else
|
|
return file;
|
|
#endif
|
|
}
|
|
|
|
gstrvec
|
|
moo_filenames_from_locale (gstrvec files)
|
|
{
|
|
gstrvec ret;
|
|
ret.reserve(files.size());
|
|
|
|
for (gstr& file: files)
|
|
{
|
|
ret.emplace_back (moo_filename_from_locale (std::move (file)));
|
|
|
|
if (ret.back().empty())
|
|
ret.erase (ret.end() - 1);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
gstr
|
|
_moo_filename_to_uri (const char* file)
|
|
{
|
|
gstr tmp;
|
|
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
|
|
if (!_moo_path_is_absolute (file))
|
|
{
|
|
gstr cd = g::get_current_dir ();
|
|
tmp = g::build_filename (cd, file);
|
|
file = tmp;
|
|
}
|
|
|
|
return g::filename_to_uri (file);
|
|
}
|
|
|
|
|
|
#ifndef __WIN32__
|
|
|
|
static gstr
|
|
normalize_path_string (const char *path)
|
|
{
|
|
GPtrArray *comps;
|
|
gboolean first_slash;
|
|
char **pieces, **p;
|
|
char *normpath;
|
|
|
|
g_return_val_if_fail (path != NULL, NULL);
|
|
|
|
first_slash = (path[0] == G_DIR_SEPARATOR);
|
|
|
|
pieces = g_strsplit (path, G_DIR_SEPARATOR_S, 0);
|
|
g_return_val_if_fail (pieces != NULL, NULL);
|
|
|
|
comps = g_ptr_array_new ();
|
|
|
|
for (p = pieces; *p != NULL; ++p)
|
|
{
|
|
char *s = *p;
|
|
gboolean push = TRUE;
|
|
gboolean pop = FALSE;
|
|
|
|
if (!strcmp (s, "") || !strcmp (s, "."))
|
|
{
|
|
push = FALSE;
|
|
}
|
|
else if (!strcmp (s, ".."))
|
|
{
|
|
if (!comps->len && first_slash)
|
|
{
|
|
push = FALSE;
|
|
}
|
|
else if (comps->len)
|
|
{
|
|
push = FALSE;
|
|
pop = TRUE;
|
|
}
|
|
}
|
|
|
|
if (pop)
|
|
{
|
|
g_free (comps->pdata[comps->len - 1]);
|
|
g_ptr_array_remove_index (comps, comps->len - 1);
|
|
}
|
|
|
|
if (push)
|
|
g_ptr_array_add (comps, g_strdup (s));
|
|
}
|
|
|
|
g_ptr_array_add (comps, NULL);
|
|
|
|
if (comps->len == 1)
|
|
{
|
|
if (first_slash)
|
|
normpath = g_strdup (G_DIR_SEPARATOR_S);
|
|
else
|
|
normpath = g_strdup (".");
|
|
}
|
|
else
|
|
{
|
|
char *tmp = g_strjoinv (G_DIR_SEPARATOR_S, (char**) comps->pdata);
|
|
|
|
if (first_slash)
|
|
{
|
|
guint len = strlen (tmp);
|
|
normpath = g_renew (char, tmp, len + 2);
|
|
memmove (normpath + 1, normpath, len + 1);
|
|
normpath[0] = G_DIR_SEPARATOR;
|
|
}
|
|
else
|
|
{
|
|
normpath = tmp;
|
|
}
|
|
}
|
|
|
|
g_strfreev (pieces);
|
|
g_strfreev ((char**) comps->pdata);
|
|
g_ptr_array_free (comps, FALSE);
|
|
|
|
return gstr::wrap_new (normpath);
|
|
}
|
|
|
|
static gstr
|
|
normalize_full_path_unix (const char *path)
|
|
{
|
|
g_return_val_if_fail (path != NULL, NULL);
|
|
|
|
gstr normpath = normalize_path_string (path);
|
|
g_return_val_if_fail (!normpath.empty(), NULL);
|
|
|
|
guint len = strlen (normpath);
|
|
|
|
if (len > 1 && normpath[len - 1] == G_DIR_SEPARATOR)
|
|
{
|
|
char* p = normpath.get_mutable ();
|
|
p[len - 1] = 0;
|
|
}
|
|
|
|
return normpath;
|
|
}
|
|
|
|
#endif /* !__WIN32__ */
|
|
|
|
static gstr
|
|
normalize_full_path_win32 (const gstr& fullpath)
|
|
{
|
|
char *prefix;
|
|
char *path;
|
|
guint slashes;
|
|
|
|
g_return_val_if_fail (!fullpath.empty(), fullpath);
|
|
|
|
if (fullpath[1] == ':')
|
|
{
|
|
prefix = g_strndup (fullpath, 3);
|
|
prefix[2] = '\\';
|
|
|
|
if (fullpath[2] == '\\')
|
|
path = g_strdup (&fullpath[3]);
|
|
else
|
|
path = g_strdup (&fullpath[2]);
|
|
}
|
|
else
|
|
{
|
|
prefix = NULL;
|
|
path = g_strdup (fullpath);
|
|
}
|
|
|
|
g_strdelimit (path, "/", '\\');
|
|
|
|
for (slashes = 0; path[slashes] == '\\'; ++slashes) ;
|
|
|
|
if (!prefix && slashes)
|
|
prefix = g_strndup (path, slashes);
|
|
|
|
if (slashes)
|
|
memmove (path, path + slashes, strlen (path) + 1 - slashes);
|
|
|
|
g_assert (prefix || *path);
|
|
g_assert (*path != '\\');
|
|
|
|
if (*path)
|
|
{
|
|
char **comps;
|
|
guint n_comps, i;
|
|
|
|
comps = g_strsplit (path, "\\", 0);
|
|
n_comps = g_strv_length (comps);
|
|
|
|
for (i = 0; i < n_comps; )
|
|
{
|
|
if (strcmp (comps[i], "") == 0 || strcmp (comps[i], ".") == 0 ||
|
|
(i == 0 && strcmp (comps[i], "..") == 0))
|
|
{
|
|
g_free (comps[i]);
|
|
MOO_ELMMOVE (comps + i, comps + i + 1, n_comps - i);
|
|
n_comps -= 1;
|
|
}
|
|
else if (strcmp (comps[i], "..") == 0)
|
|
{
|
|
g_free (comps[i-1]);
|
|
g_free (comps[i]);
|
|
MOO_ELMMOVE (comps + i - 1, comps + i + 1, n_comps - i);
|
|
n_comps -= 2;
|
|
i -= 1;
|
|
}
|
|
else
|
|
{
|
|
i += 1;
|
|
}
|
|
}
|
|
|
|
if (n_comps > 0)
|
|
{
|
|
char *tmp = g_strjoinv ("\\", comps);
|
|
g_free (path);
|
|
path = tmp;
|
|
}
|
|
else
|
|
{
|
|
path[0] = 0;
|
|
}
|
|
|
|
g_strfreev (comps);
|
|
}
|
|
|
|
if (*path)
|
|
{
|
|
if (prefix)
|
|
{
|
|
char *tmp = g_strconcat (prefix, path, NULL);
|
|
g_free (path);
|
|
path = tmp;
|
|
}
|
|
}
|
|
else if (prefix)
|
|
{
|
|
g_free (path);
|
|
path = prefix;
|
|
prefix = NULL;
|
|
}
|
|
else
|
|
{
|
|
g_free (path);
|
|
path = g_strdup (".");
|
|
}
|
|
|
|
g_free (prefix);
|
|
return gstr::wrap_new (path);
|
|
}
|
|
|
|
static char*
|
|
normalize_full_path_win32_test_func (const char* fullpath)
|
|
{
|
|
return normalize_full_path_win32 (gstr::wrap_const (fullpath)).release_owned();
|
|
}
|
|
|
|
gstr
|
|
_moo_normalize_file_path (const gstr& filename)
|
|
{
|
|
g_return_val_if_fail (!filename.empty(), gstr::null);
|
|
|
|
gstr tmp;
|
|
const gstr* to_normalize = &filename;
|
|
|
|
if (!_moo_path_is_absolute (filename))
|
|
{
|
|
gstr working_dir = g::get_current_dir ();
|
|
g_return_val_if_fail (!working_dir.empty(), filename);
|
|
tmp = g::build_filename (working_dir, filename);
|
|
to_normalize = &tmp;
|
|
}
|
|
|
|
#ifdef __WIN32__
|
|
return normalize_full_path_win32 (*to_normalize);
|
|
#else
|
|
return normalize_full_path_unix (*to_normalize);
|
|
#endif
|
|
}
|
|
|
|
char *
|
|
_moo_normalize_file_path (const char *filename)
|
|
{
|
|
return _moo_normalize_file_path (gstr::wrap_const (filename)).release_owned ();
|
|
}
|
|
|
|
gboolean
|
|
_moo_path_is_absolute (const char *path)
|
|
{
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
return g_path_is_absolute (path)
|
|
#ifdef __WIN32__
|
|
/* 'C:' is an absolute path even if glib doesn't like it */
|
|
/* This will match nonsense like 1:23:23 too, but that's not
|
|
* a valid path, and it's better to have "1:23:23" in the error
|
|
* message than "c:\some\silly\current\dir\1:23:23" */
|
|
|| path[1] == ':'
|
|
#endif
|
|
;
|
|
}
|
|
|
|
|
|
static void
|
|
test_normalize_path_one (const char *path,
|
|
const char *expected,
|
|
char *(*func) (const char*),
|
|
const char *func_name)
|
|
{
|
|
char *result;
|
|
|
|
TEST_EXPECT_WARNING (!(expected && expected[0]),
|
|
"%s(%s)", func_name, TEST_FMT_STR (path));
|
|
result = func (path);
|
|
TEST_CHECK_WARNING ();
|
|
|
|
TEST_ASSERT_STR_EQ_MSG (result, expected, "%s(%s)",
|
|
func_name, TEST_FMT_STR (path));
|
|
|
|
g_free (result);
|
|
}
|
|
|
|
static GPtrArray *
|
|
make_cases (gboolean unix_paths)
|
|
{
|
|
guint i;
|
|
char *current_dir;
|
|
GPtrArray *paths = g_ptr_array_new ();
|
|
|
|
const char *abs_files_common[] = {
|
|
NULL, NULL,
|
|
"", "",
|
|
};
|
|
|
|
const char *abs_files_unix[] = {
|
|
"/usr", "/usr",
|
|
"/usr/", "/usr",
|
|
"/usr///", "/usr",
|
|
"///usr////", "/usr",
|
|
"/", "/",
|
|
"//", "/",
|
|
"/usr/../bin/../usr", "/usr",
|
|
"/usr/bin/../bin/.", "/usr/bin",
|
|
"/..", "/",
|
|
"/../../../", "/",
|
|
"/../whatever/..", "/",
|
|
"/../whatever/.././././", "/",
|
|
};
|
|
|
|
const char *abs_files_win32[] = {
|
|
"C:", "C:\\",
|
|
"C:\\", "C:\\",
|
|
"C:\\\\", "C:\\",
|
|
"C:\\foobar", "C:\\foobar",
|
|
"C:\\foobar\\", "C:\\foobar",
|
|
"C:\\foobar\\\\\\", "C:\\foobar",
|
|
"C:/", "C:\\",
|
|
"C:/foobar", "C:\\foobar",
|
|
"C:/foobar//", "C:\\foobar",
|
|
"C:/foobar///", "C:\\foobar",
|
|
"\\\\.host\\dir\\root", "\\\\.host\\dir\\root",
|
|
"\\\\.host\\dir\\root\\", "\\\\.host\\dir\\root",
|
|
"\\\\.host\\dir\\root\\..\\..\\", "\\\\.host",
|
|
};
|
|
|
|
const char *rel_files_common[] = {
|
|
".", NULL,
|
|
"././././/", NULL,
|
|
"foobar", "foobar",
|
|
"foobar/", "foobar",
|
|
"foobar//", "foobar",
|
|
"foobar/..", NULL,
|
|
"foobar/./..", NULL,
|
|
"foobar/../", NULL,
|
|
"foobar/./../", NULL,
|
|
};
|
|
|
|
#ifndef __WIN32__
|
|
const char *rel_files_unix[] = {
|
|
"foobar/com", "foobar/com",
|
|
};
|
|
#endif
|
|
|
|
#ifdef __WIN32__
|
|
const char *rel_files_win32[] = {
|
|
"foobar/com", "foobar\\com",
|
|
".\\.\\.\\.\\\\", NULL,
|
|
"foobar\\", "foobar",
|
|
"foobar\\\\", "foobar",
|
|
"foobar\\..", NULL,
|
|
"foobar\\.\\..", NULL,
|
|
"foobar\\..\\", NULL,
|
|
"foobar\\.\\..\\", NULL,
|
|
};
|
|
#endif
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (abs_files_common); i += 2)
|
|
{
|
|
g_ptr_array_add (paths, g_strdup (abs_files_common[i]));
|
|
g_ptr_array_add (paths, g_strdup (abs_files_common[i+1]));
|
|
}
|
|
|
|
if (unix_paths)
|
|
for (i = 0; i < G_N_ELEMENTS (abs_files_unix); i += 2)
|
|
{
|
|
g_ptr_array_add (paths, g_strdup (abs_files_unix[i]));
|
|
g_ptr_array_add (paths, g_strdup (abs_files_unix[i+1]));
|
|
}
|
|
else
|
|
for (i = 0; i < G_N_ELEMENTS (abs_files_win32); i += 2)
|
|
{
|
|
g_ptr_array_add (paths, g_strdup (abs_files_win32[i]));
|
|
g_ptr_array_add (paths, g_strdup (abs_files_win32[i+1]));
|
|
}
|
|
|
|
current_dir = g_get_current_dir ();
|
|
|
|
#ifndef __WIN32__
|
|
if (unix_paths)
|
|
#endif
|
|
for (i = 0; i < G_N_ELEMENTS (rel_files_common); i += 2)
|
|
{
|
|
g_ptr_array_add (paths, g_strdup (rel_files_common[i]));
|
|
if (rel_files_common[i+1])
|
|
g_ptr_array_add (paths, g_build_filename (current_dir, rel_files_common[i+1], NULL));
|
|
else
|
|
g_ptr_array_add (paths, g_strdup (current_dir));
|
|
}
|
|
|
|
#ifndef __WIN32__
|
|
if (unix_paths)
|
|
for (i = 0; i < G_N_ELEMENTS (rel_files_unix); i += 2)
|
|
{
|
|
g_ptr_array_add (paths, g_strdup (rel_files_unix[i]));
|
|
if (rel_files_unix[i+1])
|
|
g_ptr_array_add (paths, g_build_filename (current_dir, rel_files_unix[i+1], NULL));
|
|
else
|
|
g_ptr_array_add (paths, g_strdup (current_dir));
|
|
}
|
|
#endif
|
|
|
|
#ifdef __WIN32__
|
|
for (i = 0; i < G_N_ELEMENTS (rel_files_win32); i += 2)
|
|
{
|
|
g_ptr_array_add (paths, g_strdup (rel_files_win32[i]));
|
|
if (rel_files_win32[i+1])
|
|
g_ptr_array_add (paths, g_build_filename (current_dir, rel_files_win32[i+1], NULL));
|
|
else
|
|
g_ptr_array_add (paths, g_strdup (current_dir));
|
|
}
|
|
#endif
|
|
|
|
#ifndef __WIN32__
|
|
if (unix_paths)
|
|
#endif
|
|
{
|
|
char *parent_dir = g_path_get_dirname (current_dir);
|
|
g_ptr_array_add (paths, g_strdup (".."));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup ("../"));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup ("../."));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup (".././"));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup ("..//"));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
#ifdef __WIN32__
|
|
g_ptr_array_add (paths, g_strdup ("..\\"));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup ("..\\."));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup ("..\\.\\"));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
g_ptr_array_add (paths, g_strdup ("..\\\\"));
|
|
g_ptr_array_add (paths, g_strdup (parent_dir));
|
|
#endif
|
|
g_free (parent_dir);
|
|
}
|
|
|
|
g_free (current_dir);
|
|
return paths;
|
|
}
|
|
|
|
static void
|
|
test_normalize_file_path (void)
|
|
{
|
|
GPtrArray *paths;
|
|
guint i;
|
|
|
|
#ifndef __WIN32__
|
|
paths = make_cases (TRUE);
|
|
#else
|
|
paths = make_cases (FALSE);
|
|
#endif
|
|
|
|
for (i = 0; i < paths->len; i += 2)
|
|
{
|
|
test_normalize_path_one (reinterpret_cast<char*> (paths->pdata[i]),
|
|
reinterpret_cast<char*> (paths->pdata[i+1]),
|
|
_moo_normalize_file_path,
|
|
"_moo_normalize_file_path");
|
|
g_free (paths->pdata[i]);
|
|
g_free (paths->pdata[i+1]);
|
|
}
|
|
|
|
g_ptr_array_free (paths, TRUE);
|
|
}
|
|
|
|
#ifndef __WIN32__
|
|
static void
|
|
test_normalize_file_path_win32 (void)
|
|
{
|
|
GPtrArray *paths;
|
|
guint i;
|
|
|
|
paths = make_cases (FALSE);
|
|
|
|
for (i = 0; i < paths->len; i += 2)
|
|
{
|
|
test_normalize_path_one (reinterpret_cast<char*> (paths->pdata[i]),
|
|
reinterpret_cast<char*> (paths->pdata[i+1]),
|
|
normalize_full_path_win32_test_func,
|
|
"normalize_full_path_win32");
|
|
g_free (paths->pdata[i]);
|
|
g_free (paths->pdata[i+1]);
|
|
}
|
|
|
|
g_ptr_array_free (paths, TRUE);
|
|
}
|
|
#endif
|
|
|
|
void
|
|
moo_test_mooutils_fs (void)
|
|
{
|
|
MooTestSuite *suite;
|
|
|
|
suite = moo_test_suite_new ("mooutils-fs", "mooutils/mooutils-fs.c", NULL, NULL, NULL);
|
|
|
|
moo_test_suite_add_test (suite, "_moo_normalize_file_path", "test of _moo_normalize_file_path()",
|
|
(MooTestFunc) test_normalize_file_path, NULL);
|
|
#ifndef __WIN32__
|
|
moo_test_suite_add_test (suite, "normalize_full_path_win32", "test of normalize_full_path_win32()",
|
|
(MooTestFunc) test_normalize_file_path_win32, NULL);
|
|
#endif
|
|
}
|
|
|
|
|
|
int
|
|
_moo_mkdir (const char *path, mgw_errno_t *err)
|
|
{
|
|
#ifndef __WIN32__
|
|
return mgw_mkdir (path, S_IRWXU, err);
|
|
#else
|
|
return mgw_mkdir (path, 0, err);
|
|
#endif
|
|
}
|
|
|
|
|
|
/***********************************************************************/
|
|
/* Glob matching
|
|
*/
|
|
|
|
#define MOO_GLOB_REGEX
|
|
#include <mooglib/moo-glib.h>
|
|
|
|
typedef struct _MooGlob {
|
|
#ifdef MOO_GLOB_REGEX
|
|
GRegex *re;
|
|
#else
|
|
char *pattern;
|
|
#endif
|
|
} MooGlob;
|
|
|
|
#ifdef MOO_GLOB_REGEX
|
|
static char *
|
|
glob_to_re (const char *pattern)
|
|
{
|
|
GString *string;
|
|
const char *p, *piece, *bracket;
|
|
char *escaped;
|
|
|
|
g_return_val_if_fail (pattern != NULL, NULL);
|
|
|
|
p = piece = pattern;
|
|
string = g_string_new (NULL);
|
|
|
|
while (*p)
|
|
{
|
|
switch (*p)
|
|
{
|
|
case '*':
|
|
if (p != piece)
|
|
g_string_append_len (string, piece, p - piece);
|
|
g_string_append (string, ".*");
|
|
piece = ++p;
|
|
break;
|
|
|
|
case '?':
|
|
if (p != piece)
|
|
g_string_append_len (string, piece, p - piece);
|
|
g_string_append_c (string, '.');
|
|
piece = ++p;
|
|
break;
|
|
|
|
case '[':
|
|
if (!(bracket = strchr (p + 1, ']')))
|
|
{
|
|
g_warning ("in %s: unmatched '['", pattern);
|
|
goto error;
|
|
}
|
|
|
|
if (p != piece)
|
|
g_string_append_len (string, piece, p - piece);
|
|
|
|
g_string_append_c (string, '[');
|
|
if (p[1] == '^')
|
|
{
|
|
g_string_append_c (string, '^');
|
|
escaped = g_regex_escape_string (p + 2, (int) (bracket - p - 2));
|
|
}
|
|
else
|
|
{
|
|
escaped = g_regex_escape_string (p + 1, (int) (bracket - p - 1));
|
|
}
|
|
g_string_append (string, escaped);
|
|
g_free (escaped);
|
|
g_string_append_c (string, ']');
|
|
piece = p = bracket + 1;
|
|
break;
|
|
|
|
case '\\':
|
|
case '|':
|
|
case '(':
|
|
case ')':
|
|
case ']':
|
|
case '{':
|
|
case '}':
|
|
case '^':
|
|
case '$':
|
|
case '+':
|
|
case '.':
|
|
if (p != piece)
|
|
g_string_append_len (string, piece, p - piece);
|
|
g_string_append_c (string, '\\');
|
|
g_string_append_c (string, *p);
|
|
piece = ++p;
|
|
break;
|
|
|
|
default:
|
|
p = g_utf8_next_char (p);
|
|
}
|
|
}
|
|
|
|
if (*piece)
|
|
g_string_append (string, piece);
|
|
|
|
g_string_append_c (string, '$');
|
|
|
|
if (0)
|
|
_moo_message ("converted '%s' to '%s'\n", pattern, string->str);
|
|
|
|
return g_string_free (string, FALSE);
|
|
|
|
error:
|
|
g_string_free (string, TRUE);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static MooGlob *
|
|
_moo_glob_new (const char *pattern)
|
|
{
|
|
MooGlob *gl;
|
|
GRegex *re;
|
|
char *re_pattern;
|
|
GRegexCompileFlags flags = GRegexCompileFlags (0);
|
|
GError *error = NULL;
|
|
|
|
g_return_val_if_fail (pattern != NULL, NULL);
|
|
|
|
#ifdef __WIN32__
|
|
flags = G_REGEX_CASELESS;
|
|
#endif
|
|
|
|
if (!(re_pattern = glob_to_re (pattern)))
|
|
return NULL;
|
|
|
|
re = g_regex_new (re_pattern, flags, GRegexMatchFlags (0), &error);
|
|
|
|
g_free (re_pattern);
|
|
|
|
if (!re)
|
|
{
|
|
g_warning ("%s", moo_error_message (error));
|
|
g_error_free (error);
|
|
return NULL;
|
|
}
|
|
|
|
gl = g_new0 (MooGlob, 1);
|
|
gl->re = re;
|
|
|
|
return gl;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
_moo_glob_match (MooGlob *glob,
|
|
const char *filename_utf8)
|
|
{
|
|
g_return_val_if_fail (glob != NULL, FALSE);
|
|
g_return_val_if_fail (filename_utf8 != NULL, FALSE);
|
|
g_return_val_if_fail (g_utf8_validate (filename_utf8, -1, NULL), FALSE);
|
|
|
|
return g_regex_match (glob->re, filename_utf8, GRegexMatchFlags (0), NULL);
|
|
}
|
|
#endif
|
|
|
|
|
|
static void
|
|
_moo_glob_free (MooGlob *glob)
|
|
{
|
|
if (glob)
|
|
{
|
|
#ifdef MOO_GLOB_REGEX
|
|
g_regex_unref (glob->re);
|
|
#else
|
|
g_free (glob->pattern);
|
|
#endif
|
|
g_free (glob);
|
|
}
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_glob_match_simple (const char *pattern,
|
|
const char *filename)
|
|
{
|
|
MooGlob *gl;
|
|
gboolean result = FALSE;
|
|
|
|
g_return_val_if_fail (pattern != NULL, FALSE);
|
|
g_return_val_if_fail (filename != NULL, FALSE);
|
|
|
|
if ((gl = _moo_glob_new (pattern)))
|
|
result = _moo_glob_match (gl, filename);
|
|
|
|
if (result && 0)
|
|
_moo_message ("'%s' matched '%s'", filename, pattern);
|
|
|
|
_moo_glob_free (gl);
|
|
return result;
|
|
}
|