/* GIO - GLib Input, Output and Streaming Library * * Copyright 2016 Endless Mobile, 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 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 <http://www.gnu.org/licenses/>. */ #include "config.h" #include <sys/stat.h> #include <fcntl.h> #include <errno.h> #include <string.h> #include "gdocumentportal.h" #include "xdp-dbus.h" #include "gstdio.h" #ifdef G_OS_UNIX #include "gunixfdlist.h" #endif #ifndef O_PATH #define O_PATH 0 #endif #ifndef O_CLOEXEC #define O_CLOEXEC 0 #else #define HAVE_O_CLOEXEC 1 #endif static GXdpDocuments *documents; static char *documents_mountpoint; static gboolean init_document_portal (void) { static gsize documents_inited = 0; if (g_once_init_enter (&documents_inited)) { GError *error = NULL; GDBusConnection *connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error); if (connection != NULL) { documents = gxdp_documents_proxy_new_sync (connection, 0, "org.freedesktop.portal.Documents", "/org/freedesktop/portal/documents", NULL, &error); if (documents != NULL) { gxdp_documents_call_get_mount_point_sync (documents, &documents_mountpoint, NULL, &error); if (error != NULL) { g_warning ("Cannot get document portal mount point: %s", error->message); g_error_free (error); } } else { g_warning ("Cannot create document portal proxy: %s", error->message); g_error_free (error); } g_object_unref (connection); } else { g_warning ("Cannot connect to session bus when initializing document portal: %s", error->message); g_error_free (error); } g_once_init_leave (&documents_inited, 1); } return (documents != NULL && documents_mountpoint != NULL); } char * g_document_portal_add_document (GFile *file, GError **error) { char *doc_path, *basename; char *doc_id = NULL; char *doc_uri = NULL; char *path = NULL; GUnixFDList *fd_list = NULL; int fd, fd_in, errsv; gboolean ret; if (!init_document_portal ()) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_INITIALIZED, "Document portal is not available"); goto out; } path = g_file_get_path (file); fd = g_open (path, O_PATH | O_CLOEXEC); errsv = errno; if (fd == -1) { g_set_error (error, G_IO_ERROR, g_io_error_from_errno (errsv), "Failed to open %s", path); goto out; } #ifndef HAVE_O_CLOEXEC fcntl (fd, F_SETFD, FD_CLOEXEC); #endif fd_list = g_unix_fd_list_new (); fd_in = g_unix_fd_list_append (fd_list, fd, error); g_close (fd, NULL); if (fd_in == -1) goto out; ret = gxdp_documents_call_add_sync (documents, g_variant_new_handle (fd_in), TRUE, TRUE, fd_list, &doc_id, NULL, NULL, error); if (!ret) goto out; basename = g_path_get_basename (path); doc_path = g_build_filename (documents_mountpoint, doc_id, basename, NULL); g_free (basename); doc_uri = g_filename_to_uri (doc_path, NULL, NULL); g_free (doc_path); out: if (fd_list) g_object_unref (fd_list); g_free (path); g_free (doc_id); return doc_uri; } /* Flags accepted by org.freedesktop.portal.Documents.AddFull */ enum { XDP_ADD_FLAGS_REUSE_EXISTING = (1 << 0), XDP_ADD_FLAGS_PERSISTENT = (1 << 1), XDP_ADD_FLAGS_AS_NEEDED_BY_APP = (1 << 2), XDP_ADD_FLAGS_FLAGS_ALL = ((1 << 3) - 1) }; GList * g_document_portal_add_documents (GList *uris, const char *app_id, GError **error) { int length; GList *ruris = NULL; gboolean *as_is; GVariantBuilder builder; GUnixFDList *fd_list = NULL; GList *l; gsize i, j; const char *permissions[] = { "read", "write", NULL }; char **doc_ids = NULL; GVariant *extra_out = NULL; if (!init_document_portal ()) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_INITIALIZED, "Document portal is not available"); return NULL; } length = g_list_length (uris); as_is = g_new0 (gboolean, length); g_variant_builder_init (&builder, G_VARIANT_TYPE ("ah")); fd_list = g_unix_fd_list_new (); for (l = uris, i = 0; l; l = l->next, i++) { const char *uri = l->data; int idx = -1; char *path = NULL; path = g_filename_from_uri (uri, NULL, NULL); if (path != NULL) { int fd; fd = g_open (path, O_CLOEXEC | O_PATH); if (fd >= 0) { #ifndef HAVE_O_CLOEXEC fcntl (fd, F_SETFD, FD_CLOEXEC); #endif idx = g_unix_fd_list_append (fd_list, fd, NULL); close (fd); } } g_free (path); if (idx != -1) g_variant_builder_add (&builder, "h", idx); else as_is[i] = TRUE; } if (g_unix_fd_list_get_length (fd_list) > 0) { if (!gxdp_documents_call_add_full_sync (documents, g_variant_builder_end (&builder), XDP_ADD_FLAGS_AS_NEEDED_BY_APP, app_id, permissions, fd_list, &doc_ids, &extra_out, NULL, NULL, error)) goto out; for (l = uris, i = 0, j = 0; l; l = l->next, i++) { const char *uri = l->data; char *ruri; if (as_is[i]) /* use as-is, not a file uri */ { ruri = g_strdup (uri); } else if (strcmp (doc_ids[j], "") == 0) /* not rewritten */ { ruri = g_strdup (uri); j++; } else { char *basename = g_path_get_basename (uri + strlen ("file:")); char *doc_path = g_build_filename (documents_mountpoint, doc_ids[j], basename, NULL); ruri = g_strconcat ("file:", doc_path, NULL); g_free (basename); g_free (doc_path); j++; } ruris = g_list_prepend (ruris, ruri); } ruris = g_list_reverse (ruris); } else { ruris = g_list_copy_deep (uris, (GCopyFunc)g_strdup, NULL); } out: g_clear_object (&fd_list); g_clear_pointer (&extra_out, g_variant_unref); g_clear_pointer (&doc_ids, g_strfreev); g_free (as_is); return ruris; }