Merge branch 'wip/oholy/trashing-locations' into 'master'

glocalfile: Allow trashing only for locations supported by trash:///

See merge request GNOME/glib!49
This commit is contained in:
Philip Withnall 2018-06-06 10:58:22 +00:00
commit 205059b9c0
4 changed files with 133 additions and 1 deletions

View File

@ -1769,6 +1769,7 @@ _g_local_file_has_trash_dir (const char *dirname, dev_t dir_dev)
char uid_str[32]; char uid_str[32];
GStatBuf global_stat, trash_stat; GStatBuf global_stat, trash_stat;
gboolean res; gboolean res;
GUnixMountEntry *mount;
if (g_once_init_enter (&home_dev_set)) if (g_once_init_enter (&home_dev_set))
{ {
@ -1787,6 +1788,16 @@ _g_local_file_has_trash_dir (const char *dirname, dev_t dir_dev)
if (topdir == NULL) if (topdir == NULL)
return FALSE; return FALSE;
mount = g_unix_mount_at (topdir, NULL);
if (mount == NULL || g_unix_mount_is_system_internal (mount))
{
g_clear_pointer (&mount, g_unix_mount_free);
return FALSE;
}
g_clear_pointer (&mount, g_unix_mount_free);
globaldir = g_build_filename (topdir, ".Trash", NULL); globaldir = g_build_filename (topdir, ".Trash", NULL);
if (g_lstat (globaldir, &global_stat) == 0 && if (g_lstat (globaldir, &global_stat) == 0 &&
S_ISDIR (global_stat.st_mode) && S_ISDIR (global_stat.st_mode) &&
@ -1930,6 +1941,7 @@ g_local_file_trash (GFile *file,
{ {
uid_t uid; uid_t uid;
char uid_str[32]; char uid_str[32];
GUnixMountEntry *mount;
uid = geteuid (); uid = geteuid ();
g_snprintf (uid_str, sizeof (uid_str), "%lu", (unsigned long)uid); g_snprintf (uid_str, sizeof (uid_str), "%lu", (unsigned long)uid);
@ -1942,7 +1954,21 @@ g_local_file_trash (GFile *file,
file, G_IO_ERROR_NOT_SUPPORTED); file, G_IO_ERROR_NOT_SUPPORTED);
return FALSE; return FALSE;
} }
mount = g_unix_mount_at (topdir, NULL);
if (mount == NULL || g_unix_mount_is_system_internal (mount))
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
_("Trashing on system internal mounts is not supported"));
g_clear_pointer (&mount, g_unix_mount_free);
g_free (topdir);
return FALSE;
}
g_clear_pointer (&mount, g_unix_mount_free);
/* Try looking for global trash dir $topdir/.Trash/$uid */ /* Try looking for global trash dir $topdir/.Trash/$uid */
globaldir = g_build_filename (topdir, ".Trash", NULL); globaldir = g_build_filename (topdir, ".Trash", NULL);
if (g_lstat (globaldir, &global_stat) == 0 && if (g_lstat (globaldir, &global_stat) == 0 &&

View File

@ -273,6 +273,7 @@ test_programs += \
unix-mounts \ unix-mounts \
unix-streams \ unix-streams \
g-file-info-filesystem-readonly \ g-file-info-filesystem-readonly \
trash \
$(NULL) $(NULL)
test_extra_programs += \ test_extra_programs += \

View File

@ -123,6 +123,7 @@ if host_machine.system() != 'windows'
'unix-streams', 'unix-streams',
'g-file-info-filesystem-readonly', 'g-file-info-filesystem-readonly',
'gschema-compile', 'gschema-compile',
'trash',
] ]
# Uninstalled because of the check-for-executable logic in DesktopAppInfo # Uninstalled because of the check-for-executable logic in DesktopAppInfo

104
gio/tests/trash.c Normal file
View File

@ -0,0 +1,104 @@
/*
* Copyright (C) 2018 Red Hat, Inc.
*
* This library 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
* licence, or (at your option) any later version.
*
* This 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 Lesser General Public
* License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include <glib.h>
#ifndef G_OS_UNIX
#error This is a Unix-specific test
#endif
#include <glib/gstdio.h>
#include <gio/gio.h>
#include <gio/gunixmounts.h>
/* Test that g_file_trash() returns G_IO_ERROR_NOT_SUPPORTED for files on system mounts. */
static void
test_trash_not_supported (void)
{
GFile *file;
GFileIOStream *stream;
GUnixMountEntry *mount;
GFileInfo *info;
GError *error = NULL;
gboolean ret;
GStatBuf file_stat, home_stat;
/* The test assumes that tmp file is located on system internal mount. */
file = g_file_new_tmp ("test-trashXXXXXX", &stream, &error);
g_assert_no_error (error);
g_assert_cmpint (g_lstat (g_file_peek_path (file), &file_stat), ==, 0);
g_test_message ("File: %s (dev: %lu)", g_file_peek_path (file), file_stat.st_dev);
g_assert_cmpint (g_stat (g_get_home_dir (), &home_stat), ==, 0);
g_test_message ("Home: %s (dev: %lu)", g_get_home_dir (), home_stat.st_dev);
if (file_stat.st_dev == home_stat.st_dev)
{
g_test_skip ("The file has to be on another filesystem than the home folder to run this test");
g_object_unref (stream);
g_object_unref (file);
return;
}
mount = g_unix_mount_for (g_file_peek_path (file), NULL);
g_assert_nonnull (mount);
g_assert_true (g_unix_mount_is_system_internal (mount));
g_unix_mount_free (mount);
/* g_file_trash() shouldn't be supported on system internal mounts,
* because those are not monitored by gvfsd-trash.
*/
ret = g_file_trash (file, NULL, &error);
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
g_test_message ("Error: %s", error->message);
g_assert_false (ret);
g_clear_error (&error);
info = g_file_query_info (file,
G_FILE_ATTRIBUTE_ACCESS_CAN_TRASH,
G_FILE_QUERY_INFO_NONE,
NULL,
&error);
g_assert_no_error (error);
g_assert_false (g_file_info_get_attribute_boolean (info,
G_FILE_ATTRIBUTE_ACCESS_CAN_TRASH));
g_io_stream_close (G_IO_STREAM (stream), NULL, &error);
g_assert_no_error (error);
g_object_unref (info);
g_object_unref (stream);
g_object_unref (file);
}
int
main (int argc, char *argv[])
{
g_test_init (&argc, &argv, NULL);
g_test_bug_base ("htps://gitlab.gnome.org/GNOME/glib/issues/");
g_test_bug ("251");
g_test_add_func ("/trash/not-supported", test_trash_not_supported);
return g_test_run ();
}