/* * Copyright © 2024 GNOME Foundation Inc. * * SPDX-License-Identifier: LGPL-2.1-or-later * * 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 License, or (at your option) any later version. * * This library 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 . * * Authors: Julian Sparber * Philip Withnall */ /* A stub implementation of xdg-desktop-portal */ #ifdef __FreeBSD__ #include #include #include #endif /* __FreeBSD__ */ #include #include #include #include "fake-desktop-portal.h" #include "fake-openuri-portal-generated.h" #include "fake-request-portal-generated.h" struct _GFakeDesktopPortalThread { GObject parent_instance; char *address; /* (not nullable) */ GCancellable *cancellable; /* (not nullable) (owned) */ GThread *thread; /* (not nullable) (owned) */ GCond cond; /* (mutex mutex) */ GMutex mutex; gboolean ready; /* (mutex mutex) */ char *request_activation_token; /* (mutex mutex) */ char *request_uri; /* (mutex mutex) */ } FakeDesktopPortalThread; G_DEFINE_FINAL_TYPE (GFakeDesktopPortalThread, g_fake_desktop_portal_thread, G_TYPE_OBJECT) static void g_fake_desktop_portal_thread_finalize (GObject *object); static void g_fake_desktop_portal_thread_class_init (GFakeDesktopPortalThreadClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->finalize = g_fake_desktop_portal_thread_finalize; } static void g_fake_desktop_portal_thread_init (GFakeDesktopPortalThread *self) { self->cancellable = g_cancellable_new (); g_cond_init (&self->cond); g_mutex_init (&self->mutex); } static void g_fake_desktop_portal_thread_finalize (GObject *object) { GFakeDesktopPortalThread *self = G_FAKE_DESKTOP_PORTAL_THREAD (object); g_assert (self->thread == NULL); /* should already have been joined */ g_mutex_clear (&self->mutex); g_cond_clear (&self->cond); g_clear_object (&self->cancellable); g_clear_pointer (&self->address, g_free); g_clear_pointer (&self->request_activation_token, g_free); g_clear_pointer (&self->request_uri, g_free); G_OBJECT_CLASS (g_fake_desktop_portal_thread_parent_class)->finalize (object); } static gboolean on_handle_close (FakeRequest *object, GDBusMethodInvocation *invocation, gpointer user_data) { g_test_message ("Got close request"); fake_request_complete_close (object, invocation); return G_DBUS_METHOD_INVOCATION_HANDLED; } static char* get_request_path (GDBusMethodInvocation *invocation, const char *token) { char *request_obj_path; char *sender; sender = g_strdup (g_dbus_method_invocation_get_sender (invocation) + 1); /* The object path needs to be the specific format. * See: https://flatpak.github.io/xdg-desktop-portal/docs/doc-org.freedesktop.portal.Request.html#org-freedesktop-portal-request */ for (size_t i = 0; sender[i]; i++) if (sender[i] == '.') sender[i] = '_'; request_obj_path = g_strdup_printf ("/org/freedesktop/portal/desktop/request/%s/%s", sender, token); g_free (sender); return request_obj_path; } static gboolean handle_request (GFakeDesktopPortalThread *self, FakeOpenURI *object, GDBusMethodInvocation *invocation, const gchar *arg_parent_window, const gchar *arg_uri, gboolean open_file, GVariant *arg_options) { const char *activation_token = NULL; GError *error = NULL; FakeRequest *interface_request; GVariantBuilder opt_builder; char *request_obj_path; const char *token = NULL; if (arg_options) { g_variant_lookup (arg_options, "activation_token", "&s", &activation_token); g_variant_lookup (arg_options, "handle_token", "&s", &token); } g_set_str (&self->request_activation_token, activation_token); g_set_str (&self->request_uri, arg_uri); request_obj_path = get_request_path (invocation, token ? token : "t"); if (open_file) { g_test_message ("Got open file request for %s", arg_uri); fake_open_uri_complete_open_file (object, invocation, NULL, request_obj_path); } else { g_test_message ("Got open URI request for %s", arg_uri); fake_open_uri_complete_open_uri (object, invocation, request_obj_path); } interface_request = fake_request_skeleton_new (); g_variant_builder_init (&opt_builder, G_VARIANT_TYPE_VARDICT); g_signal_connect (interface_request, "handle-close", G_CALLBACK (on_handle_close), NULL); g_dbus_interface_skeleton_export (G_DBUS_INTERFACE_SKELETON (interface_request), g_dbus_method_invocation_get_connection (invocation), request_obj_path, &error); g_assert_no_error (error); g_dbus_interface_skeleton_set_flags (G_DBUS_INTERFACE_SKELETON (interface_request), G_DBUS_INTERFACE_SKELETON_FLAGS_HANDLE_METHOD_INVOCATIONS_IN_THREAD); g_test_message ("Request skeleton exported at %s", request_obj_path); /* We can't use `fake_request_emit_response()` because it doesn't set the sender */ g_dbus_connection_emit_signal (g_dbus_method_invocation_get_connection (invocation), g_dbus_method_invocation_get_sender (invocation), request_obj_path, "org.freedesktop.portal.Request", "Response", g_variant_new ("(u@a{sv})", 0, /* Success */ g_variant_builder_end (&opt_builder)), NULL); g_test_message ("Response emitted"); g_dbus_interface_skeleton_unexport (G_DBUS_INTERFACE_SKELETON (interface_request)); g_free (request_obj_path); g_object_unref (interface_request); return G_DBUS_METHOD_INVOCATION_HANDLED; } /* This is currently private as there’s only one user of it, but it could become * a public API in future. */ static char * _g_fd_query_path (int fd, GError **error) { #if defined(__FreeBSD__) struct kinfo_file kf; kf.kf_structsize = sizeof (kf); if (fcntl (fd, F_KINFO, &kf) < 0) { int saved_errno = errno; g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, "Error querying file information for FD %d: %s", fd, g_strerror (saved_errno)); return NULL; } return g_strdup (kf.kf_path); #elif defined(G_OS_UNIX) char *path = NULL; char *proc_path = g_strdup_printf ("/proc/self/fd/%d", fd); path = g_file_read_link (proc_path, error); g_free (proc_path); return g_steal_pointer (&path); #else /* - A NetBSD implementation would probably use `fcntl()` with `F_GETPATH`: * https://man.netbsd.org/fcntl.2 * - A Windows implementation would probably use `GetFinalPathNameByHandleW()`: * https://learn.microsoft.com/en-us/windows/win32/api/fileapi/nf-fileapi-getfinalpathnamebyhandlea * - A Hurd implementation could open("/dev/fd/%u"): * https://gitlab.gnome.org/GNOME/glib/-/merge_requests/4396#note_2279923 */ #error "_g_fd_query_path() not supported on this platform" #endif } static char * handle_to_uri (GVariant *handle, GUnixFDList *fd_list) { int fd = -1; int fd_id; char *path; char *uri; GError *local_error = NULL; fd_id = g_variant_get_handle (handle); fd = g_unix_fd_list_get (fd_list, fd_id, NULL); if (fd == -1) return NULL; path = _g_fd_query_path (fd, &local_error); g_assert_no_error (local_error); uri = g_filename_to_uri (path, NULL, NULL); g_free (path); close (fd); return uri; } static gboolean on_handle_open_file (FakeOpenURI *object, GDBusMethodInvocation *invocation, GUnixFDList *fd_list, const gchar *arg_parent_window, GVariant *arg_fd, GVariant *arg_options, gpointer user_data) { GFakeDesktopPortalThread *self = G_FAKE_DESKTOP_PORTAL_THREAD (user_data); char *uri = NULL; uri = handle_to_uri (arg_fd, fd_list); handle_request (self, object, invocation, arg_parent_window, uri, TRUE, arg_options); g_free (uri); return G_DBUS_METHOD_INVOCATION_HANDLED; } static gboolean on_handle_open_uri (FakeOpenURI *object, GDBusMethodInvocation *invocation, const gchar *arg_parent_window, const gchar *arg_uri, GVariant *arg_options, gpointer user_data) { GFakeDesktopPortalThread *self = G_FAKE_DESKTOP_PORTAL_THREAD (user_data); handle_request (self, object, invocation, arg_parent_window, arg_uri, TRUE, arg_options); return G_DBUS_METHOD_INVOCATION_HANDLED; } static void on_name_acquired (GDBusConnection *connection, const gchar *name, gpointer user_data) { GFakeDesktopPortalThread *self = G_FAKE_DESKTOP_PORTAL_THREAD (user_data); g_test_message ("Acquired the name %s", name); g_mutex_lock (&self->mutex); self->ready = TRUE; g_cond_signal (&self->cond); g_mutex_unlock (&self->mutex); } static void on_name_lost (GDBusConnection *connection, const gchar *name, gpointer user_data) { g_test_message ("Lost the name %s", name); } static gboolean cancelled_cb (GCancellable *cancellable, gpointer user_data) { g_test_message ("fake-desktop-portal cancelled"); return G_SOURCE_CONTINUE; } static gpointer fake_desktop_portal_thread_cb (gpointer user_data) { GFakeDesktopPortalThread *self = G_FAKE_DESKTOP_PORTAL_THREAD (user_data); GMainContext *context = NULL; GDBusConnection *connection = NULL; GSource *source = NULL; guint id; FakeOpenURI *interface_open_uri; GError *local_error = NULL; context = g_main_context_new (); g_main_context_push_thread_default (context); connection = g_dbus_connection_new_for_address_sync (self->address, G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT | G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION, NULL, self->cancellable, &local_error); g_assert_no_error (local_error); /* Listen for cancellation. The source will wake up the context iteration * which can then re-check its exit condition below. */ source = g_cancellable_source_new (self->cancellable); g_source_set_callback (source, G_SOURCE_FUNC (cancelled_cb), NULL, NULL); g_source_attach (source, context); g_source_unref (source); /* Set up the interface */ g_test_message ("Acquired a message bus connection"); interface_open_uri = fake_open_uri_skeleton_new (); g_signal_connect (interface_open_uri, "handle-open-file", G_CALLBACK (on_handle_open_file), self); g_signal_connect (interface_open_uri, "handle-open-uri", G_CALLBACK (on_handle_open_uri), self); g_dbus_interface_skeleton_export (G_DBUS_INTERFACE_SKELETON (interface_open_uri), connection, "/org/freedesktop/portal/desktop", &local_error); g_assert_no_error (local_error); /* Own the portal name */ id = g_bus_own_name_on_connection (connection, "org.freedesktop.portal.Desktop", G_BUS_NAME_OWNER_FLAGS_ALLOW_REPLACEMENT | G_BUS_NAME_OWNER_FLAGS_REPLACE, on_name_acquired, on_name_lost, self, NULL); while (!g_cancellable_is_cancelled (self->cancellable)) g_main_context_iteration (context, TRUE); g_bus_unown_name (id); g_clear_object (&connection); g_dbus_interface_skeleton_unexport (G_DBUS_INTERFACE_SKELETON (interface_open_uri)); g_object_unref (interface_open_uri); g_main_context_pop_thread_default (context); g_clear_pointer (&context, g_main_context_unref); return NULL; } /* Get the activation token given to the most recent OpenURI request * * Returns: (transfer none) (nullable: an activation token */ const gchar * g_fake_desktop_portal_thread_get_last_request_activation_token (GFakeDesktopPortalThread *self) { g_return_val_if_fail (G_IS_FAKE_DESKTOP_PORTAL_THREAD (self), NULL); return self->request_activation_token; } /* Get the file or URI given to the most recent OpenURI request * * Returns: (transfer none) (nullable): an URI */ const gchar * g_fake_desktop_portal_thread_get_last_request_uri (GFakeDesktopPortalThread *self) { g_return_val_if_fail (G_IS_FAKE_DESKTOP_PORTAL_THREAD (self), NULL); return self->request_uri; } /* * Create a new #GFakeDesktopPortalThread. The thread isn’t started until * g_fake_desktop_portal_thread_run() is called on it. * * Returns: (transfer full): the new fake desktop portal wrapper */ GFakeDesktopPortalThread * g_fake_desktop_portal_thread_new (const char *address) { GFakeDesktopPortalThread *self = g_object_new (G_TYPE_FAKE_DESKTOP_PORTAL_THREAD, NULL); self->address = g_strdup (address); return g_steal_pointer (&self); } /* * Start a worker thread which will run a fake * `org.freedesktop.portal.Desktops` portal on the bus at @address. This is * intended to be used with #GTestDBus to mock up a portal from within a unit * test process, rather than relying on D-Bus activation of a mock portal * subprocess. * * It blocks until the thread has owned its D-Bus name and is ready to handle * requests. */ void g_fake_desktop_portal_thread_run (GFakeDesktopPortalThread *self) { g_return_if_fail (G_IS_FAKE_DESKTOP_PORTAL_THREAD (self)); g_return_if_fail (self->thread == NULL); self->thread = g_thread_new ("fake-desktop-portal", fake_desktop_portal_thread_cb, self); /* Block until the thread is ready. */ g_mutex_lock (&self->mutex); while (!self->ready) g_cond_wait (&self->cond, &self->mutex); g_mutex_unlock (&self->mutex); } /* Stop and join a worker thread started with fake_desktop_portal_thread_run(). * Blocks until the thread has stopped and joined. * * Once this has been called, it’s safe to drop the final reference on @self. */ void g_fake_desktop_portal_thread_stop (GFakeDesktopPortalThread *self) { g_return_if_fail (G_IS_FAKE_DESKTOP_PORTAL_THREAD (self)); g_return_if_fail (self->thread != NULL); g_cancellable_cancel (self->cancellable); g_thread_join (g_steal_pointer (&self->thread)); } /* Whether fake-desktop-portal is supported on this platform. This basically * means whether _g_fd_query_path() will work at runtime. */ gboolean g_fake_desktop_portal_is_supported (void) { #ifdef __GNU__ return FALSE; #else return TRUE; #endif }