5068ac
From af4d0d88604af7c196e461a743f2d1e81239d76a Mon Sep 17 00:00:00 2001
5068ac
From: Ondrej Holy <oholy@redhat.com>
5068ac
Date: Tue, 14 May 2019 09:31:37 +0200
5068ac
Subject: [PATCH 2/3] daemon: Handle lockdown option to disable writing
5068ac
5068ac
Handle the new mount-removable-storage-devices-as-read-only option of
5068ac
org.gnome.desktop.lockdown schema and present AFC, MTP, GPhoto2 devices
5068ac
as read-only if enabled.
5068ac
---
5068ac
 daemon/gvfsbackend.c           | 62 ++++++++++++++++++++++++++++++++--
5068ac
 daemon/gvfsbackend.h           |  6 ++++
5068ac
 daemon/gvfsbackendafc.c        |  2 ++
5068ac
 daemon/gvfsbackendgphoto2.c    |  1 +
5068ac
 daemon/gvfsbackendmtp.c        |  1 +
5068ac
 daemon/gvfsjobcopy.c           |  7 ++++
5068ac
 daemon/gvfsjobdelete.c         |  7 ++++
5068ac
 daemon/gvfsjobmakedirectory.c  |  7 ++++
5068ac
 daemon/gvfsjobmakesymlink.c    |  7 ++++
5068ac
 daemon/gvfsjobmove.c           |  7 ++++
5068ac
 daemon/gvfsjobopenforwrite.c   |  7 ++++
5068ac
 daemon/gvfsjobpush.c           |  7 ++++
5068ac
 daemon/gvfsjobqueryfsinfo.c    | 11 ++----
5068ac
 daemon/gvfsjobsetattribute.c   |  7 ++++
5068ac
 daemon/gvfsjobsetdisplayname.c |  7 ++++
5068ac
 daemon/gvfsjobtrash.c          |  7 ++++
5068ac
 16 files changed, 143 insertions(+), 10 deletions(-)
5068ac
5068ac
diff --git a/daemon/gvfsbackend.c b/daemon/gvfsbackend.c
5068ac
index 4fd3455c..599733ef 100644
5068ac
--- a/daemon/gvfsbackend.c
5068ac
+++ b/daemon/gvfsbackend.c
5068ac
@@ -80,6 +80,9 @@ struct _GVfsBackendPrivate
5068ac
   char *default_location;
5068ac
   GMountSpec *mount_spec;
5068ac
   gboolean block_requests;
5068ac
+
5068ac
+  GSettings *lockdown_settings;
5068ac
+  gboolean readonly_lockdown;
5068ac
 };
5068ac
 
5068ac
 
5068ac
@@ -155,7 +158,9 @@ g_vfs_backend_finalize (GObject *object)
5068ac
   g_free (backend->priv->default_location);
5068ac
   if (backend->priv->mount_spec)
5068ac
     g_mount_spec_unref (backend->priv->mount_spec);
5068ac
-  
5068ac
+
5068ac
+  g_clear_object (&backend->priv->lockdown_settings);
5068ac
+
5068ac
   if (G_OBJECT_CLASS (g_vfs_backend_parent_class)->finalize)
5068ac
     (*G_OBJECT_CLASS (g_vfs_backend_parent_class)->finalize) (object);
5068ac
 }
5068ac
@@ -587,7 +592,29 @@ g_vfs_backend_add_auto_info (GVfsBackend *backend,
5068ac
        g_file_attribute_matcher_matches (matcher,
5068ac
                                          G_FILE_ATTRIBUTE_THUMBNAILING_FAILED)))
5068ac
     get_thumbnail_attributes (uri, info);
5068ac
-  
5068ac
+
5068ac
+  if (backend->priv->readonly_lockdown)
5068ac
+    {
5068ac
+      g_file_info_set_attribute_boolean (info, G_FILE_ATTRIBUTE_ACCESS_CAN_DELETE, FALSE);
5068ac
+      g_file_info_set_attribute_boolean (info, G_FILE_ATTRIBUTE_ACCESS_CAN_RENAME, FALSE);
5068ac
+      g_file_info_set_attribute_boolean (info, G_FILE_ATTRIBUTE_ACCESS_CAN_TRASH, FALSE);
5068ac
+      g_file_info_set_attribute_boolean (info, G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE, FALSE);
5068ac
+    }
5068ac
+}
5068ac
+
5068ac
+void
5068ac
+g_vfs_backend_add_auto_fs_info (GVfsBackend *backend,
5068ac
+                                GFileAttributeMatcher *matcher,
5068ac
+                                GFileInfo *info)
5068ac
+{
5068ac
+  const char *type;
5068ac
+
5068ac
+  type = g_vfs_backend_get_backend_type (backend);
5068ac
+  if (type)
5068ac
+    g_file_info_set_attribute_string (info, G_FILE_ATTRIBUTE_GVFS_BACKEND, type);
5068ac
+
5068ac
+  if (backend->priv->readonly_lockdown)
5068ac
+    g_file_info_set_attribute_boolean (info, G_FILE_ATTRIBUTE_FILESYSTEM_READONLY, TRUE);
5068ac
 }
5068ac
 
5068ac
 void
5068ac
@@ -1047,3 +1074,34 @@ g_vfs_backend_force_unmount (GVfsBackend *backend)
5068ac
                                   (GAsyncReadyCallback) forced_unregister_mount_callback,
5068ac
                                   NULL);
5068ac
 }
5068ac
+
5068ac
+static void
5068ac
+lockdown_settings_changed (GSettings *settings,
5068ac
+                           gchar     *key,
5068ac
+                           gpointer   user_data)
5068ac
+{
5068ac
+  GVfsBackend *backend = G_VFS_BACKEND (user_data);
5068ac
+
5068ac
+  backend->priv->readonly_lockdown = g_settings_get_boolean (settings,
5068ac
+                                                             "mount-removable-storage-devices-as-read-only");
5068ac
+}
5068ac
+
5068ac
+
5068ac
+void
5068ac
+g_vfs_backend_handle_readonly_lockdown (GVfsBackend *backend)
5068ac
+{
5068ac
+  backend->priv->lockdown_settings = g_settings_new ("org.gnome.desktop.lockdown");
5068ac
+  backend->priv->readonly_lockdown = g_settings_get_boolean (backend->priv->lockdown_settings,
5068ac
+                                                             "mount-removable-storage-devices-as-read-only");
5068ac
+  g_signal_connect_object (backend->priv->lockdown_settings,
5068ac
+                           "changed",
5068ac
+                           G_CALLBACK (lockdown_settings_changed),
5068ac
+                           backend,
5068ac
+                           0);
5068ac
+}
5068ac
+
5068ac
+gboolean
5068ac
+g_vfs_backend_get_readonly_lockdown (GVfsBackend *backend)
5068ac
+{
5068ac
+  return backend->priv->readonly_lockdown;
5068ac
+}
5068ac
diff --git a/daemon/gvfsbackend.h b/daemon/gvfsbackend.h
5068ac
index 9c7476cf..431dd290 100644
5068ac
--- a/daemon/gvfsbackend.h
5068ac
+++ b/daemon/gvfsbackend.h
5068ac
@@ -516,6 +516,9 @@ void        g_vfs_backend_add_auto_info                  (GVfsBackend
5068ac
 							  GFileAttributeMatcher *matcher,
5068ac
 							  GFileInfo             *info,
5068ac
 							  const char            *uri);
5068ac
+void        g_vfs_backend_add_auto_fs_info               (GVfsBackend           *backend,
5068ac
+                                                          GFileAttributeMatcher *matcher,
5068ac
+                                                          GFileInfo             *info);
5068ac
 
5068ac
 void        g_vfs_backend_set_block_requests             (GVfsBackend           *backend,
5068ac
                                                           gboolean               value);
5068ac
@@ -534,6 +537,9 @@ gboolean    g_vfs_backend_invocation_first_handler       (GVfsDBusMount *object,
5068ac
                                                           GDBusMethodInvocation *invocation,
5068ac
                                                           GVfsBackend *backend);
5068ac
 
5068ac
+void        g_vfs_backend_handle_readonly_lockdown       (GVfsBackend *backend);
5068ac
+gboolean    g_vfs_backend_get_readonly_lockdown          (GVfsBackend *backend);
5068ac
+
5068ac
 G_END_DECLS
5068ac
 
5068ac
 #endif /* __G_VFS_BACKEND_H__ */
5068ac
diff --git a/daemon/gvfsbackendafc.c b/daemon/gvfsbackendafc.c
5068ac
index b6e6a106..ce68aa45 100644
5068ac
--- a/daemon/gvfsbackendafc.c
5068ac
+++ b/daemon/gvfsbackendafc.c
5068ac
@@ -2760,6 +2760,8 @@ g_vfs_backend_afc_init (GVfsBackendAfc *self)
5068ac
     }
5068ac
 
5068ac
   g_mutex_init (&self->apps_lock);
5068ac
+
5068ac
+  g_vfs_backend_handle_readonly_lockdown (G_VFS_BACKEND (self));
5068ac
 }
5068ac
 
5068ac
 static void
5068ac
diff --git a/daemon/gvfsbackendgphoto2.c b/daemon/gvfsbackendgphoto2.c
5068ac
index 51e9a3bd..7e50194a 100644
5068ac
--- a/daemon/gvfsbackendgphoto2.c
5068ac
+++ b/daemon/gvfsbackendgphoto2.c
5068ac
@@ -614,6 +614,7 @@ g_vfs_backend_gphoto2_init (GVfsBackendGphoto2 *gphoto2_backend)
5068ac
   g_mutex_init (&gphoto2_backend->lock);
5068ac
 
5068ac
   g_vfs_backend_set_display_name (backend, "gphoto2");
5068ac
+  g_vfs_backend_handle_readonly_lockdown (G_VFS_BACKEND (backend));
5068ac
 
5068ac
   mount_spec = g_mount_spec_new ("gphoto2");
5068ac
   g_vfs_backend_set_mount_spec (backend, mount_spec);
5068ac
diff --git a/daemon/gvfsbackendmtp.c b/daemon/gvfsbackendmtp.c
5068ac
index e3a25ef2..c4f1e855 100644
5068ac
--- a/daemon/gvfsbackendmtp.c
5068ac
+++ b/daemon/gvfsbackendmtp.c
5068ac
@@ -379,6 +379,7 @@ g_vfs_backend_mtp_init (GVfsBackendMtp *backend)
5068ac
   g_mutex_init (&backend->mutex);
5068ac
   g_vfs_backend_set_display_name (G_VFS_BACKEND (backend), "mtp");
5068ac
   g_vfs_backend_set_icon_name (G_VFS_BACKEND (backend), "multimedia-player");
5068ac
+  g_vfs_backend_handle_readonly_lockdown (G_VFS_BACKEND (backend));
5068ac
 
5068ac
   mount_spec = g_mount_spec_new ("mtp");
5068ac
   g_vfs_backend_set_mount_spec (G_VFS_BACKEND (backend), mount_spec);
5068ac
diff --git a/daemon/gvfsjobcopy.c b/daemon/gvfsjobcopy.c
5068ac
index 785d7480..cf33da56 100644
5068ac
--- a/daemon/gvfsjobcopy.c
5068ac
+++ b/daemon/gvfsjobcopy.c
5068ac
@@ -141,6 +141,13 @@ try (GVfsJob *job)
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
   gboolean res;
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_copy == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobdelete.c b/daemon/gvfsjobdelete.c
5068ac
index 92892f15..8d5e5b8e 100644
5068ac
--- a/daemon/gvfsjobdelete.c
5068ac
+++ b/daemon/gvfsjobdelete.c
5068ac
@@ -120,6 +120,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobDelete *op_job = G_VFS_JOB_DELETE (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_delete == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobmakedirectory.c b/daemon/gvfsjobmakedirectory.c
5068ac
index 98bb28d5..56a9c42a 100644
5068ac
--- a/daemon/gvfsjobmakedirectory.c
5068ac
+++ b/daemon/gvfsjobmakedirectory.c
5068ac
@@ -120,6 +120,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobMakeDirectory *op_job = G_VFS_JOB_MAKE_DIRECTORY (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_make_directory == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobmakesymlink.c b/daemon/gvfsjobmakesymlink.c
5068ac
index 2c55e26b..2684b6fd 100644
5068ac
--- a/daemon/gvfsjobmakesymlink.c
5068ac
+++ b/daemon/gvfsjobmakesymlink.c
5068ac
@@ -124,6 +124,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobMakeSymlink *op_job = G_VFS_JOB_MAKE_SYMLINK (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_make_symlink == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobmove.c b/daemon/gvfsjobmove.c
5068ac
index cc4ad220..5903d17a 100644
5068ac
--- a/daemon/gvfsjobmove.c
5068ac
+++ b/daemon/gvfsjobmove.c
5068ac
@@ -141,6 +141,13 @@ try (GVfsJob *job)
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
   gboolean res;
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_move == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobopenforwrite.c b/daemon/gvfsjobopenforwrite.c
5068ac
index 68eae532..60ce64f9 100644
5068ac
--- a/daemon/gvfsjobopenforwrite.c
5068ac
+++ b/daemon/gvfsjobopenforwrite.c
5068ac
@@ -230,6 +230,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobOpenForWrite *op_job = G_VFS_JOB_OPEN_FOR_WRITE (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (op_job->mode == OPEN_FOR_WRITE_CREATE)
5068ac
     {
5068ac
       if (class->try_create == NULL)
5068ac
diff --git a/daemon/gvfsjobpush.c b/daemon/gvfsjobpush.c
5068ac
index d7e48d86..a8df73a8 100644
5068ac
--- a/daemon/gvfsjobpush.c
5068ac
+++ b/daemon/gvfsjobpush.c
5068ac
@@ -146,6 +146,13 @@ try (GVfsJob *job)
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
   gboolean res;
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_push == NULL)
5068ac
     return FALSE;
5068ac
 
5068ac
diff --git a/daemon/gvfsjobqueryfsinfo.c b/daemon/gvfsjobqueryfsinfo.c
5068ac
index 898052ea..3363311a 100644
5068ac
--- a/daemon/gvfsjobqueryfsinfo.c
5068ac
+++ b/daemon/gvfsjobqueryfsinfo.c
5068ac
@@ -147,15 +147,10 @@ create_reply (GVfsJob *job,
5068ac
               GDBusMethodInvocation *invocation)
5068ac
 {
5068ac
   GVfsJobQueryFsInfo *op_job = G_VFS_JOB_QUERY_FS_INFO (job);
5068ac
-  const char *type;
5068ac
-
5068ac
-  type = g_vfs_backend_get_backend_type (op_job->backend);
5068ac
-
5068ac
-  if (type)
5068ac
-    g_file_info_set_attribute_string (op_job->file_info,
5068ac
-				      G_FILE_ATTRIBUTE_GVFS_BACKEND,
5068ac
-				      type);
5068ac
 
5068ac
+  g_vfs_backend_add_auto_fs_info (op_job->backend,
5068ac
+                                  op_job->attribute_matcher,
5068ac
+                                  op_job->file_info);
5068ac
   g_file_info_set_attribute_mask (op_job->file_info,
5068ac
                                   op_job->attribute_matcher);
5068ac
 
5068ac
diff --git a/daemon/gvfsjobsetattribute.c b/daemon/gvfsjobsetattribute.c
5068ac
index 1efe7c94..ac7618a4 100644
5068ac
--- a/daemon/gvfsjobsetattribute.c
5068ac
+++ b/daemon/gvfsjobsetattribute.c
5068ac
@@ -146,6 +146,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobSetAttribute *op_job = G_VFS_JOB_SET_ATTRIBUTE (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_set_attribute == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobsetdisplayname.c b/daemon/gvfsjobsetdisplayname.c
5068ac
index badb10dd..e12ae879 100644
5068ac
--- a/daemon/gvfsjobsetdisplayname.c
5068ac
+++ b/daemon/gvfsjobsetdisplayname.c
5068ac
@@ -124,6 +124,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobSetDisplayName *op_job = G_VFS_JOB_SET_DISPLAY_NAME (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_set_display_name == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
diff --git a/daemon/gvfsjobtrash.c b/daemon/gvfsjobtrash.c
5068ac
index 1738f8a2..5234ebf8 100644
5068ac
--- a/daemon/gvfsjobtrash.c
5068ac
+++ b/daemon/gvfsjobtrash.c
5068ac
@@ -119,6 +119,13 @@ try (GVfsJob *job)
5068ac
   GVfsJobTrash *op_job = G_VFS_JOB_TRASH (job);
5068ac
   GVfsBackendClass *class = G_VFS_BACKEND_GET_CLASS (op_job->backend);
5068ac
 
5068ac
+  if (g_vfs_backend_get_readonly_lockdown (op_job->backend))
5068ac
+    {
5068ac
+      g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
5068ac
+                        _("Filesystem is read-only"));
5068ac
+      return TRUE;
5068ac
+    }
5068ac
+
5068ac
   if (class->try_trash == NULL)
5068ac
     return FALSE;
5068ac
   
5068ac
-- 
5068ac
2.21.0
5068ac