/* vi:set et sw=2 sts=2 cin cino=t0,f0,(0,{s,>2s,n-s,^-s,e-s:
* Copyright © 2018 Red Hat, Inc
*
* This program 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:
* Matthias Clasen
*/
#include "config.h"
#include
#include
#include
#include
#include
#include
#include "libglnx.h"
#include "flatpak-permission-dbus-generated.h"
#include "flatpak-builtins.h"
#include "flatpak-builtins-utils.h"
#include "flatpak-table-printer.h"
#include "flatpak-utils-private.h"
#include "flatpak-run-private.h"
static GOptionEntry options[] = {
{ NULL }
};
static char **
get_ids_for_table (XdpDbusPermissionStore *store,
const char *table)
{
char **ids = NULL;
xdp_dbus_permission_store_call_list_sync (store, table, &ids, NULL, NULL);
return ids;
}
static gboolean
list_table (XdpDbusPermissionStore *store,
const char *table,
const char *id,
FlatpakTablePrinter *printer,
GError **error)
{
const char *one_id[2];
char **ids;
int i;
if (id)
{
one_id[0] = id;
one_id[1] = NULL;
ids = (char **) one_id;
}
else
{
if (!xdp_dbus_permission_store_call_list_sync (store, table, &ids, NULL, error))
return FALSE;
}
for (i = 0; ids[i]; i++)
{
g_autoptr(GVariant) permissions = NULL;
g_autoptr(GVariant) data = NULL;
g_autoptr(GVariant) d = NULL;
g_autofree char *txt = NULL;
GVariantIter iter;
char *key;
GVariantIter *val;
if (!xdp_dbus_permission_store_call_lookup_sync (store, table, ids[i], &permissions, &data, NULL, error))
return FALSE;
d = g_variant_get_child_value (data, 0);
txt = g_variant_print (d, FALSE);
if (g_variant_iter_init (&iter, permissions) == 0)
{
flatpak_table_printer_add_column (printer, table);
flatpak_table_printer_add_column (printer, ids[i]);
flatpak_table_printer_add_column (printer, "");
flatpak_table_printer_add_column (printer, "");
flatpak_table_printer_add_column (printer, txt);
flatpak_table_printer_finish_row (printer);
}
while (g_variant_iter_loop (&iter, "{sas}", &key, &val))
{
char *p;
flatpak_table_printer_add_column (printer, table);
flatpak_table_printer_add_column (printer, ids[i]);
flatpak_table_printer_add_column (printer, key);
flatpak_table_printer_add_column (printer, "");
while (g_variant_iter_loop (val, "s", &p))
{
flatpak_table_printer_append_with_comma (printer, p);
}
flatpak_table_printer_add_column (printer, txt);
flatpak_table_printer_finish_row (printer);
}
}
return TRUE;
}
gboolean
flatpak_builtin_permission_list (int argc, char **argv,
GCancellable *cancellable,
GError **error)
{
g_autoptr(GOptionContext) context = NULL;
g_autoptr(GDBusConnection) session_bus = NULL;
XdpDbusPermissionStore *store = NULL;
const char *table;
const char *id;
g_autoptr(FlatpakTablePrinter) printer = NULL;
context = g_option_context_new (_("[TABLE] [ID] - List permissions"));
g_option_context_set_translation_domain (context, GETTEXT_PACKAGE);
if (!flatpak_option_context_parse (context, options, &argc, &argv,
FLATPAK_BUILTIN_FLAG_NO_DIR,
NULL, cancellable, error))
return FALSE;
if (argc < 2)
table = NULL;
else
table = argv[1];
if (argc < 3)
id = NULL;
else
id = argv[2];
if (argc > 3)
return usage_error (context, _("Too many arguments"), error);
session_bus = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, error);
if (session_bus == NULL)
return FALSE;
store = xdp_dbus_permission_store_proxy_new_sync (session_bus, 0,
"org.freedesktop.impl.portal.PermissionStore",
"/org/freedesktop/impl/portal/PermissionStore",
NULL, error);
if (store == NULL)
return FALSE;
printer = flatpak_table_printer_new ();
int i;
i = 0;
flatpak_table_printer_set_column_title (printer, i++, _("Table"));
flatpak_table_printer_set_column_title (printer, i++, _("Object"));
flatpak_table_printer_set_column_title (printer, i++, _("App"));
flatpak_table_printer_set_column_title (printer, i++, _("Permissions"));
flatpak_table_printer_set_column_title (printer, i++, _("Data"));
if (table)
{
if (!list_table (store, table, id, printer, error))
return FALSE;
}
else
{
g_auto(GStrv) tables = get_permission_tables (store);
for (i = 0; tables[i]; i++)
{
if (!list_table (store, tables[i], NULL, printer, error))
return FALSE;
}
}
flatpak_table_printer_print (printer);
return TRUE;
}
gboolean
flatpak_complete_permission_list (FlatpakCompletion *completion)
{
g_autoptr(GOptionContext) context = NULL;
g_autoptr(GDBusConnection) session_bus = NULL;
XdpDbusPermissionStore *store = NULL;
int i;
context = g_option_context_new ("");
if (!flatpak_option_context_parse (context, options, &completion->argc, &completion->argv,
FLATPAK_BUILTIN_FLAG_NO_DIR, NULL, NULL, NULL))
return FALSE;
session_bus = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, NULL);
if (session_bus == NULL)
return FALSE;
store = xdp_dbus_permission_store_proxy_new_sync (session_bus, 0,
"org.freedesktop.impl.portal.PermissionStore",
"/org/freedesktop/impl/portal/PermissionStore",
NULL, NULL);
if (store == NULL)
return FALSE;
switch (completion->argc)
{
case 0:
case 1: /* TABLE */
flatpak_complete_options (completion, global_entries);
flatpak_complete_options (completion, options);
{
g_auto(GStrv) tables = get_permission_tables (store);
for (i = 0; tables != NULL && tables[i] != NULL; i++)
{
flatpak_complete_word (completion, "%s ", tables[i]);
}
}
break;
case 2:
{
g_auto(GStrv) ids = get_ids_for_table (store, completion->argv[1]);
for (i = 0; ids != NULL && ids[i] != NULL; i++)
{
flatpak_complete_word (completion, "%s ", ids[i]);
}
}
break;
}
return TRUE;
}