Blame gio/gopenuriportal.c

Packit Service d3d246
/* GIO - GLib Input, Output and Streaming Library
Packit Service d3d246
 *
Packit Service d3d246
 * Copyright 2017 Red Hat, Inc.
Packit Service d3d246
 *
Packit Service d3d246
 * This library is free software; you can redistribute it and/or
Packit Service d3d246
 * modify it under the terms of the GNU Lesser General Public
Packit Service d3d246
 * License as published by the Free Software Foundation; either
Packit Service d3d246
 * version 2.1 of the License, or (at your option) any later version.
Packit Service d3d246
 *
Packit Service d3d246
 * This library is distributed in the hope that it will be useful,
Packit Service d3d246
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service d3d246
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service d3d246
 * Lesser General Public License for more details.
Packit Service d3d246
 *
Packit Service d3d246
 * You should have received a copy of the GNU Lesser General Public
Packit Service d3d246
 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit Service d3d246
 */
Packit Service d3d246
Packit Service d3d246
#include "config.h"
Packit Service d3d246
Packit Service d3d246
#include <sys/stat.h>
Packit Service d3d246
#include <fcntl.h>
Packit Service d3d246
#include <errno.h>
Packit Service d3d246
#include <string.h>
Packit Service d3d246
Packit Service d3d246
#include "gopenuriportal.h"
Packit Service d3d246
#include "xdp-dbus.h"
Packit Service d3d246
#include "gstdio.h"
Packit Service d3d246
Packit Service d3d246
#ifdef G_OS_UNIX
Packit Service d3d246
#include "gunixfdlist.h"
Packit Service d3d246
#endif
Packit Service d3d246
Packit Service d3d246
#ifndef O_PATH
Packit Service d3d246
#define O_PATH 0
Packit Service d3d246
#endif
Packit Service d3d246
#ifndef O_CLOEXEC
Packit Service d3d246
#define O_CLOEXEC 0
Packit Service d3d246
#else
Packit Service d3d246
#define HAVE_O_CLOEXEC 1
Packit Service d3d246
#endif
Packit Service d3d246
Packit Service d3d246
Packit Service d3d246
static GXdpOpenURI *openuri;
Packit Service d3d246
Packit Service d3d246
static gboolean
Packit Service d3d246
init_openuri_portal (void)
Packit Service d3d246
{
Packit Service d3d246
  static gsize openuri_inited = 0;
Packit Service d3d246
Packit Service d3d246
  if (g_once_init_enter (&openuri_inited))
Packit Service d3d246
    {
Packit Service d3d246
      GError *error = NULL;
Packit Service d3d246
      GDBusConnection *connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
Packit Service d3d246
Packit Service d3d246
      if (connection != NULL)
Packit Service d3d246
        {
Packit Service d3d246
          openuri = gxdp_open_uri_proxy_new_sync (connection, 0,
Packit Service d3d246
                                                  "org.freedesktop.portal.Desktop",
Packit Service d3d246
                                                  "/org/freedesktop/portal/desktop",
Packit Service d3d246
                                                  NULL, &error);
Packit Service d3d246
          if (openuri == NULL)
Packit Service d3d246
            {
Packit Service d3d246
              g_warning ("Cannot create document portal proxy: %s", error->message);
Packit Service d3d246
              g_error_free (error);
Packit Service d3d246
            }
Packit Service d3d246
Packit Service d3d246
          g_object_unref (connection);
Packit Service d3d246
        }
Packit Service d3d246
      else
Packit Service d3d246
        {
Packit Service d3d246
          g_warning ("Cannot connect to session bus when initializing document portal: %s",
Packit Service d3d246
                     error->message);
Packit Service d3d246
          g_error_free (error);
Packit Service d3d246
        }
Packit Service d3d246
Packit Service d3d246
      g_once_init_leave (&openuri_inited, 1);
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  return openuri != NULL;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
gboolean
Packit Service d3d246
g_openuri_portal_open_uri (const char  *uri,
Packit Service d3d246
                           const char  *parent_window,
Packit Service d3d246
                           GError     **error)
Packit Service d3d246
{
Packit Service d3d246
  GFile *file = NULL;
Packit Service d3d246
  GVariantBuilder opt_builder;
Packit Service d3d246
  gboolean res;
Packit Service d3d246
Packit Service d3d246
  if (!init_openuri_portal ())
Packit Service d3d246
    {
Packit Service d3d246
      g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_INITIALIZED,
Packit Service d3d246
                   "OpenURI portal is not available");
Packit Service d3d246
      return FALSE;
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  g_variant_builder_init (&opt_builder, G_VARIANT_TYPE_VARDICT);
Packit Service d3d246
Packit Service d3d246
  file = g_file_new_for_uri (uri);
Packit Service d3d246
  if (g_file_is_native (file))
Packit Service d3d246
    {
Packit Service d3d246
      char *path = NULL;
Packit Service d3d246
      GUnixFDList *fd_list = NULL;
Packit Service d3d246
      int fd, fd_id, errsv;
Packit Service d3d246
Packit Service d3d246
      path = g_file_get_path (file);
Packit Service d3d246
Packit Service d3d246
      fd = g_open (path, O_PATH | O_CLOEXEC);
Packit Service d3d246
      errsv = errno;
Packit Service d3d246
      if (fd == -1)
Packit Service d3d246
        {
Packit Service d3d246
          g_set_error (error, G_IO_ERROR, g_io_error_from_errno (errsv),
Packit Service d3d246
                       "Failed to open '%s'", path);
Packit Service d3d246
          return FALSE;
Packit Service d3d246
        }
Packit Service d3d246
Packit Service d3d246
#ifndef HAVE_O_CLOEXEC
Packit Service d3d246
      fcntl (fd, F_SETFD, FD_CLOEXEC);
Packit Service d3d246
#endif
Packit Service d3d246
      fd_list = g_unix_fd_list_new_from_array (&fd, 1);
Packit Service d3d246
      fd = -1;
Packit Service d3d246
      fd_id = 0;
Packit Service d3d246
Packit Service d3d246
      res = gxdp_open_uri_call_open_file_sync (openuri,
Packit Service d3d246
                                               parent_window ? parent_window : "",
Packit Service d3d246
                                               g_variant_new ("h", fd_id),
Packit Service d3d246
                                               g_variant_builder_end (&opt_builder),
Packit Service d3d246
                                               fd_list,
Packit Service d3d246
                                               NULL,
Packit Service d3d246
                                               NULL,
Packit Service d3d246
                                               NULL,
Packit Service d3d246
                                               error);
Packit Service d3d246
      g_free (path);
Packit Service d3d246
      g_object_unref (fd_list);
Packit Service d3d246
    }
Packit Service d3d246
  else
Packit Service d3d246
    {
Packit Service d3d246
      res = gxdp_open_uri_call_open_uri_sync (openuri,
Packit Service d3d246
                                              parent_window ? parent_window : "",
Packit Service d3d246
                                              uri,
Packit Service d3d246
                                              g_variant_builder_end (&opt_builder),
Packit Service d3d246
                                              NULL,
Packit Service d3d246
                                              NULL,
Packit Service d3d246
                                              error);
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  g_object_unref (file);
Packit Service d3d246
Packit Service d3d246
  return res;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
enum {
Packit Service d3d246
  XDG_DESKTOP_PORTAL_SUCCESS   = 0,
Packit Service d3d246
  XDG_DESKTOP_PORTAL_CANCELLED = 1,
Packit Service d3d246
  XDG_DESKTOP_PORTAL_FAILED    = 2
Packit Service d3d246
};
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
response_received (GDBusConnection *connection,
Packit Service d3d246
                   const char      *sender_name,
Packit Service d3d246
                   const char      *object_path,
Packit Service d3d246
                   const char      *interface_name,
Packit Service d3d246
                   const char      *signal_name,
Packit Service d3d246
                   GVariant        *parameters,
Packit Service d3d246
                   gpointer         user_data)
Packit Service d3d246
{
Packit Service d3d246
  GTask *task = user_data;
Packit Service d3d246
  guint32 response;
Packit Service d3d246
  guint signal_id;
Packit Service d3d246
Packit Service d3d246
  signal_id = GPOINTER_TO_UINT (g_object_get_data (G_OBJECT (task), "signal-id"));
Packit Service d3d246
  g_dbus_connection_signal_unsubscribe (connection, signal_id);
Packit Service d3d246
Packit Service d3d246
  g_variant_get (parameters, "(u@a{sv})", &response, NULL);
Packit Service d3d246
Packit Service d3d246
  switch (response)
Packit Service d3d246
    {
Packit Service d3d246
    case XDG_DESKTOP_PORTAL_SUCCESS:
Packit Service d3d246
      g_task_return_boolean (task, TRUE);
Packit Service d3d246
      break;
Packit Service d3d246
    case XDG_DESKTOP_PORTAL_CANCELLED:
Packit Service d3d246
      g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_CANCELLED, "Launch cancelled");
Packit Service d3d246
      break;
Packit Service d3d246
    case XDG_DESKTOP_PORTAL_FAILED:
Packit Service d3d246
    default:
Packit Service d3d246
      g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_FAILED, "Launch failed");
Packit Service d3d246
      break;
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  g_object_unref (task);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
open_call_done (GObject      *source,
Packit Service d3d246
                GAsyncResult *result,
Packit Service d3d246
                gpointer      user_data)
Packit Service d3d246
{
Packit Service d3d246
  GXdpOpenURI *openuri = GXDP_OPEN_URI (source);
Packit Service d3d246
  GDBusConnection *connection;
Packit Service d3d246
  GTask *task = user_data;
Packit Service d3d246
  GError *error = NULL;
Packit Service d3d246
  gboolean open_file;
Packit Service d3d246
  gboolean res;
Packit Service d3d246
  char *path;
Packit Service d3d246
  const char *handle;
Packit Service d3d246
  guint signal_id;
Packit Service d3d246
Packit Service d3d246
  connection = g_dbus_proxy_get_connection (G_DBUS_PROXY (openuri));
Packit Service d3d246
  open_file = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (task), "open-file"));
Packit Service d3d246
Packit Service d3d246
  if (open_file)
Packit Service d3d246
    res = gxdp_open_uri_call_open_file_finish (openuri, &path, NULL, result, &error);
Packit Service d3d246
  else
Packit Service d3d246
    res = gxdp_open_uri_call_open_uri_finish (openuri, &path, result, &error);
Packit Service d3d246
Packit Service d3d246
  if (!res)
Packit Service d3d246
    {
Packit Service d3d246
      g_task_return_error (task, error);
Packit Service d3d246
      g_object_unref (task);
Packit Service d3d246
      g_free (path);
Packit Service d3d246
      return;
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  handle = (const char *)g_object_get_data (G_OBJECT (task), "handle");
Packit Service d3d246
  if (strcmp (handle, path) != 0)
Packit Service d3d246
    {
Packit Service d3d246
      signal_id = GPOINTER_TO_UINT (g_object_get_data (G_OBJECT (task), "signal-id"));
Packit Service d3d246
      g_dbus_connection_signal_unsubscribe (connection, signal_id);
Packit Service d3d246
Packit Service d3d246
      signal_id = g_dbus_connection_signal_subscribe (connection,
Packit Service d3d246
                                                      "org.freedesktop.portal.Desktop",
Packit Service d3d246
                                                      "org.freedesktop.portal.Request",
Packit Service d3d246
                                                      "Response",
Packit Service d3d246
                                                      path,
Packit Service d3d246
                                                      NULL,
Packit Service d3d246
                                                      G_DBUS_SIGNAL_FLAGS_NO_MATCH_RULE,
Packit Service d3d246
                                                      response_received,
Packit Service d3d246
                                                      task,
Packit Service d3d246
                                                      NULL);
Packit Service d3d246
      g_object_set_data (G_OBJECT (task), "signal-id", GINT_TO_POINTER (signal_id));
Packit Service d3d246
    }
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
void
Packit Service d3d246
g_openuri_portal_open_uri_async (const char          *uri,
Packit Service d3d246
                                 const char          *parent_window,
Packit Service d3d246
                                 GCancellable        *cancellable,
Packit Service d3d246
                                 GAsyncReadyCallback  callback,
Packit Service d3d246
                                 gpointer             user_data)
Packit Service d3d246
{
Packit Service d3d246
  GDBusConnection *connection;
Packit Service d3d246
  GTask *task;
Packit Service d3d246
  GFile *file;
Packit Service d3d246
  GVariant *opts = NULL;
Packit Service d3d246
  int i;
Packit Service d3d246
  guint signal_id;
Packit Service d3d246
Packit Service d3d246
  if (!init_openuri_portal ())
Packit Service d3d246
    {
Packit Service d3d246
      g_task_report_new_error (NULL, callback, user_data, NULL,
Packit Service d3d246
                               G_IO_ERROR, G_IO_ERROR_NOT_INITIALIZED,
Packit Service d3d246
                               "OpenURI portal is not available");
Packit Service d3d246
      return;
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  connection = g_dbus_proxy_get_connection (G_DBUS_PROXY (openuri));
Packit Service d3d246
Packit Service d3d246
  if (callback)
Packit Service d3d246
    {
Packit Service d3d246
      GVariantBuilder opt_builder;
Packit Service d3d246
      char *token;
Packit Service d3d246
      char *sender;
Packit Service d3d246
      char *handle;
Packit Service d3d246
Packit Service d3d246
      task = g_task_new (NULL, cancellable, callback, user_data);
Packit Service d3d246
Packit Service d3d246
      token = g_strdup_printf ("gio%d", g_random_int_range (0, G_MAXINT));
Packit Service d3d246
      sender = g_strdup (g_dbus_connection_get_unique_name (connection) + 1);
Packit Service d3d246
      for (i = 0; sender[i]; i++)
Packit Service d3d246
        if (sender[i] == '.')
Packit Service d3d246
          sender[i] = '_';
Packit Service d3d246
Packit Service d3d246
      handle = g_strdup_printf ("/org/fredesktop/portal/desktop/request/%s/%s", sender, token);
Packit Service d3d246
      g_object_set_data_full (G_OBJECT (task), "handle", handle, g_free);
Packit Service d3d246
      g_free (sender);
Packit Service d3d246
Packit Service d3d246
      signal_id = g_dbus_connection_signal_subscribe (connection,
Packit Service d3d246
                                                      "org.freedesktop.portal.Desktop",
Packit Service d3d246
                                                      "org.freedesktop.portal.Request",
Packit Service d3d246
                                                      "Response",
Packit Service d3d246
                                                      handle,
Packit Service d3d246
                                                      NULL,
Packit Service d3d246
                                                      G_DBUS_SIGNAL_FLAGS_NO_MATCH_RULE,
Packit Service d3d246
                                                      response_received,
Packit Service d3d246
                                                      task,
Packit Service d3d246
                                                      NULL);
Packit Service d3d246
      g_object_set_data (G_OBJECT (task), "signal-id", GINT_TO_POINTER (signal_id));
Packit Service d3d246
Packit Service d3d246
      g_variant_builder_init (&opt_builder, G_VARIANT_TYPE_VARDICT);
Packit Service d3d246
      g_variant_builder_add (&opt_builder, "{sv}", "handle_token", g_variant_new_string (token));
Packit Service d3d246
      g_free (token);
Packit Service d3d246
Packit Service d3d246
      opts = g_variant_builder_end (&opt_builder);
Packit Service d3d246
    }
Packit Service d3d246
  else
Packit Service d3d246
    task = NULL;
Packit Service d3d246
Packit Service d3d246
  file = g_file_new_for_uri (uri);
Packit Service d3d246
  if (g_file_is_native (file))
Packit Service d3d246
    {
Packit Service d3d246
      char *path = NULL;
Packit Service d3d246
      GUnixFDList *fd_list = NULL;
Packit Service d3d246
      int fd, fd_id, errsv;
Packit Service d3d246
Packit Service d3d246
      if (task)
Packit Service d3d246
        g_object_set_data (G_OBJECT (task), "open-file", GINT_TO_POINTER (TRUE));
Packit Service d3d246
Packit Service d3d246
      path = g_file_get_path (file);
Packit Service d3d246
      fd = g_open (path, O_PATH | O_CLOEXEC);
Packit Service d3d246
      errsv = errno;
Packit Service d3d246
      if (fd == -1)
Packit Service d3d246
        {
Packit Service d3d246
          g_task_report_new_error (NULL, callback, user_data, NULL,
Packit Service d3d246
                                   G_IO_ERROR, g_io_error_from_errno (errsv),
Packit Service d3d246
                                   "OpenURI portal is not available");
Packit Service d3d246
          return;
Packit Service d3d246
        }
Packit Service d3d246
Packit Service d3d246
#ifndef HAVE_O_CLOEXEC
Packit Service d3d246
      fcntl (fd, F_SETFD, FD_CLOEXEC);
Packit Service d3d246
#endif
Packit Service d3d246
      fd_list = g_unix_fd_list_new_from_array (&fd, 1);
Packit Service d3d246
      fd = -1;
Packit Service d3d246
      fd_id = 0;
Packit Service d3d246
Packit Service d3d246
      gxdp_open_uri_call_open_file (openuri,
Packit Service d3d246
                                    parent_window ? parent_window : "",
Packit Service d3d246
                                    g_variant_new ("h", fd_id),
Packit Service d3d246
                                    opts,
Packit Service d3d246
                                    fd_list,
Packit Service d3d246
                                    cancellable,
Packit Service d3d246
                                    task ? open_call_done : NULL,
Packit Service d3d246
                                    task);
Packit Service d3d246
      g_object_unref (fd_list);
Packit Service d3d246
      g_free (path);
Packit Service d3d246
    }
Packit Service d3d246
  else
Packit Service d3d246
    {
Packit Service d3d246
      gxdp_open_uri_call_open_uri (openuri,
Packit Service d3d246
                                   parent_window ? parent_window : "",
Packit Service d3d246
                                   uri,
Packit Service d3d246
                                   opts,
Packit Service d3d246
                                   cancellable,
Packit Service d3d246
                                   task ? open_call_done : NULL,
Packit Service d3d246
                                   task);
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  g_object_unref (file);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
gboolean
Packit Service d3d246
g_openuri_portal_open_uri_finish (GAsyncResult  *result,
Packit Service d3d246
                                  GError       **error)
Packit Service d3d246
{
Packit Service d3d246
  return g_task_propagate_boolean (G_TASK (result), error);
Packit Service d3d246
}