Merge pull request #10 from endlessm/T24024-sync-methods

T24024 libeos-parental-controls: Add synchronous versions of methods
This commit is contained in:
Philip Withnall 2018-11-01 12:53:28 +00:00 committed by GitHub
commit 60c58fcba3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 261 additions and 317 deletions

View File

@ -37,28 +37,9 @@ def __get_app_filter(user_id, interactive):
If `interactive` is `True`, interactive polkit authorisation dialogues will If `interactive` is `True`, interactive polkit authorisation dialogues will
be allowed. An exception will be raised on failure.""" be allowed. An exception will be raised on failure."""
app_filter = None return EosParentalControls.get_app_filter(
exception = None
def __get_cb(obj, result, user_data):
nonlocal app_filter, exception
try:
app_filter = EosParentalControls.get_app_filter_finish(result)
except Exception as e:
exception = e
EosParentalControls.get_app_filter_async(
connection=None, user_id=user_id, connection=None, user_id=user_id,
allow_interactive_authorization=interactive, cancellable=None, allow_interactive_authorization=interactive, cancellable=None)
callback=__get_cb, user_data=None)
context = GLib.MainContext.default()
while not app_filter and not exception:
context.iteration(True)
if exception:
raise exception
return app_filter
def __get_app_filter_or_error(user_id, interactive): def __get_app_filter_or_error(user_id, interactive):
@ -77,28 +58,9 @@ def __set_app_filter(user_id, app_filter, interactive):
If `interactive` is `True`, interactive polkit authorisation dialogues will If `interactive` is `True`, interactive polkit authorisation dialogues will
be allowed. An exception will be raised on failure.""" be allowed. An exception will be raised on failure."""
finished = False EosParentalControls.set_app_filter(
exception = None
def __set_cb(obj, result, user_data):
nonlocal finished, exception
try:
EosParentalControls.set_app_filter_finish(result)
finished = True
except Exception as e:
exception = e
EosParentalControls.set_app_filter_async(
connection=None, user_id=user_id, app_filter=app_filter, connection=None, user_id=user_id, app_filter=app_filter,
allow_interactive_authorization=interactive, cancellable=None, allow_interactive_authorization=interactive, cancellable=None)
callback=__set_cb, user_data=None)
context = GLib.MainContext.default()
while not finished and not exception:
context.iteration(True)
if exception:
raise exception
def __set_app_filter_or_error(user_id, app_filter, interactive): def __set_app_filter_or_error(user_id, app_filter, interactive):

View File

@ -302,17 +302,124 @@ bus_error_to_app_filter_error (const GError *bus_error,
return g_error_copy (bus_error); return g_error_copy (bus_error);
} }
static void get_app_filter_get_bus_cb (GObject *obj, /**
GAsyncResult *result, * epc_get_app_filter:
gpointer user_data); * @connection: (nullable): a #GDBusConnection to the system bus, or %NULL to
static void get_app_filter (GDBusConnection *connection, * use the default
GTask *task); * @user_id: ID of the user to query, typically coming from getuid()
static void get_app_filter_cb (GObject *obj, * @allow_interactive_authorization: %TRUE to allow interactive polkit
GAsyncResult *result, * authorization dialogues to be displayed during the call; %FALSE otherwise
gpointer user_data); * @cancellable: (nullable): a #GCancellable, or %NULL
* @error: return location for a #GError, or %NULL
*
* Synchronous version of epc_get_app_filter_async().
*
* Returns: (transfer full): app filter for the queried user
* Since: 0.1.0
*/
EpcAppFilter *
epc_get_app_filter (GDBusConnection *connection,
uid_t user_id,
gboolean allow_interactive_authorization,
GCancellable *cancellable,
GError **error)
{
g_autofree gchar *object_path = NULL;
g_autoptr(GVariant) result_variant = NULL;
g_autoptr(GVariant) properties = NULL;
g_autoptr(GError) local_error = NULL;
g_autoptr(EpcAppFilter) app_filter = NULL;
gboolean is_whitelist;
g_auto(GStrv) app_list = NULL;
const gchar *content_rating_kind;
g_autoptr(GVariant) oars_variant = NULL;
g_autoptr(GHashTable) oars_map = NULL;
g_return_val_if_fail (connection == NULL || G_IS_DBUS_CONNECTION (connection), NULL);
g_return_val_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable), NULL);
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
if (connection == NULL)
connection = g_bus_get_sync (G_BUS_TYPE_SYSTEM, cancellable, error);
if (connection == NULL)
return NULL;
object_path = g_strdup_printf ("/org/freedesktop/Accounts/User%u", user_id);
result_variant =
g_dbus_connection_call_sync (connection,
"org.freedesktop.Accounts",
object_path,
"org.freedesktop.DBus.Properties",
"GetAll",
g_variant_new ("(s)", "com.endlessm.ParentalControls.AppFilter"),
G_VARIANT_TYPE ("(a{sv})"),
allow_interactive_authorization
? G_DBUS_CALL_FLAGS_ALLOW_INTERACTIVE_AUTHORIZATION
: G_DBUS_CALL_FLAGS_NONE,
-1, /* timeout, ms */
cancellable,
&local_error);
if (local_error != NULL)
{
g_autoptr(GError) app_filter_error = bus_error_to_app_filter_error (local_error,
user_id);
g_propagate_error (error, g_steal_pointer (&app_filter_error));
return NULL;
}
/* Extract the properties we care about. They may be silently omitted from the
* results if we dont have permission to access them. */
properties = g_variant_get_child_value (result_variant, 0);
if (!g_variant_lookup (properties, "app-filter", "(b^as)",
&is_whitelist, &app_list))
{
g_set_error (error, EPC_APP_FILTER_ERROR,
EPC_APP_FILTER_ERROR_PERMISSION_DENIED,
_("Not allowed to query app filter data for user %u"),
user_id);
return NULL;
}
if (!g_variant_lookup (properties, "oars-filter", "(&s@a{ss})",
&content_rating_kind, &oars_variant))
{
/* Default value. */
content_rating_kind = "oars-1.1";
oars_variant = g_variant_new ("@a{ss} {}");
}
/* Check that the OARS filter is in a format we support. Currently, thats
* only oars-1.0 and oars-1.1. */
if (!g_str_equal (content_rating_kind, "oars-1.0") &&
!g_str_equal (content_rating_kind, "oars-1.1"))
{
g_set_error (error, EPC_APP_FILTER_ERROR,
EPC_APP_FILTER_ERROR_INVALID_DATA,
_("OARS filter for user %u has an unrecognized kind %s"),
user_id, content_rating_kind);
return NULL;
}
/* Success. Create an #EpcAppFilter object to contain the results. */
app_filter = g_new0 (EpcAppFilter, 1);
app_filter->ref_count = 1;
app_filter->user_id = user_id;
app_filter->app_list = g_steal_pointer (&app_list);
app_filter->app_list_type =
is_whitelist ? EPC_APP_FILTER_LIST_WHITELIST : EPC_APP_FILTER_LIST_BLACKLIST;
app_filter->oars_ratings = g_steal_pointer (&oars_variant);
return g_steal_pointer (&app_filter);
}
static void get_app_filter_thread_cb (GTask *task,
gpointer source_object,
gpointer task_data,
GCancellable *cancellable);
typedef struct typedef struct
{ {
GDBusConnection *connection; /* (nullable) (owned) */
uid_t user_id; uid_t user_id;
gboolean allow_interactive_authorization; gboolean allow_interactive_authorization;
} GetAppFilterData; } GetAppFilterData;
@ -320,6 +427,7 @@ typedef struct
static void static void
get_app_filter_data_free (GetAppFilterData *data) get_app_filter_data_free (GetAppFilterData *data)
{ {
g_clear_object (&data->connection);
g_free (data); g_free (data);
} }
@ -332,8 +440,8 @@ G_DEFINE_AUTOPTR_CLEANUP_FUNC (GetAppFilterData, get_app_filter_data_free)
* @user_id: ID of the user to query, typically coming from getuid() * @user_id: ID of the user to query, typically coming from getuid()
* @allow_interactive_authorization: %TRUE to allow interactive polkit * @allow_interactive_authorization: %TRUE to allow interactive polkit
* authorization dialogues to be displayed during the call; %FALSE otherwise * authorization dialogues to be displayed during the call; %FALSE otherwise
* @callback: a #GAsyncReadyCallback
* @cancellable: (nullable): a #GCancellable, or %NULL * @cancellable: (nullable): a #GCancellable, or %NULL
* @callback: a #GAsyncReadyCallback
* @user_data: user data to pass to @callback * @user_data: user data to pass to @callback
* *
* Asynchronously get a snapshot of the app filter settings for the given * Asynchronously get a snapshot of the app filter settings for the given
@ -356,7 +464,6 @@ epc_get_app_filter_async (GDBusConnection *connection,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data) gpointer user_data)
{ {
g_autoptr(GDBusConnection) connection_owned = NULL;
g_autoptr(GTask) task = NULL; g_autoptr(GTask) task = NULL;
g_autoptr(GetAppFilterData) data = NULL; g_autoptr(GetAppFilterData) data = NULL;
@ -367,134 +474,34 @@ epc_get_app_filter_async (GDBusConnection *connection,
g_task_set_source_tag (task, epc_get_app_filter_async); g_task_set_source_tag (task, epc_get_app_filter_async);
data = g_new0 (GetAppFilterData, 1); data = g_new0 (GetAppFilterData, 1);
data->connection = (connection != NULL) ? g_object_ref (connection) : NULL;
data->user_id = user_id; data->user_id = user_id;
data->allow_interactive_authorization = allow_interactive_authorization; data->allow_interactive_authorization = allow_interactive_authorization;
g_task_set_task_data (task, g_steal_pointer (&data), g_task_set_task_data (task, g_steal_pointer (&data),
(GDestroyNotify) get_app_filter_data_free); (GDestroyNotify) get_app_filter_data_free);
if (connection == NULL) g_task_run_in_thread (task, get_app_filter_thread_cb);
g_bus_get (G_BUS_TYPE_SYSTEM, cancellable,
get_app_filter_get_bus_cb, g_steal_pointer (&task));
else
get_app_filter (connection, g_steal_pointer (&task));
} }
static void static void
get_app_filter_get_bus_cb (GObject *obj, get_app_filter_thread_cb (GTask *task,
GAsyncResult *result, gpointer source_object,
gpointer user_data) gpointer task_data,
GCancellable *cancellable)
{ {
g_autoptr(GTask) task = G_TASK (user_data); g_autoptr(EpcAppFilter) filter = NULL;
g_autoptr(GDBusConnection) connection = NULL; GetAppFilterData *data = task_data;
g_autoptr(GError) local_error = NULL; g_autoptr(GError) local_error = NULL;
connection = g_bus_get_finish (result, &local_error); filter = epc_get_app_filter (data->connection, data->user_id,
data->allow_interactive_authorization,
cancellable, &local_error);
if (local_error != NULL) if (local_error != NULL)
g_task_return_error (task, g_steal_pointer (&local_error)); g_task_return_error (task, g_steal_pointer (&local_error));
else else
get_app_filter (connection, g_steal_pointer (&task)); g_task_return_pointer (task, g_steal_pointer (&filter),
} (GDestroyNotify) epc_app_filter_unref);
static void
get_app_filter (GDBusConnection *connection,
GTask *task)
{
g_autofree gchar *object_path = NULL;
GCancellable *cancellable;
GetAppFilterData *data = g_task_get_task_data (task);
cancellable = g_task_get_cancellable (task);
object_path = g_strdup_printf ("/org/freedesktop/Accounts/User%u",
data->user_id);
g_dbus_connection_call (connection,
"org.freedesktop.Accounts",
object_path,
"org.freedesktop.DBus.Properties",
"GetAll",
g_variant_new ("(s)", "com.endlessm.ParentalControls.AppFilter"),
G_VARIANT_TYPE ("(a{sv})"),
data->allow_interactive_authorization
? G_DBUS_CALL_FLAGS_ALLOW_INTERACTIVE_AUTHORIZATION
: G_DBUS_CALL_FLAGS_NONE,
-1, /* timeout, ms */
cancellable,
get_app_filter_cb,
g_steal_pointer (&task));
}
static void
get_app_filter_cb (GObject *obj,
GAsyncResult *result,
gpointer user_data)
{
GDBusConnection *connection = G_DBUS_CONNECTION (obj);
g_autoptr(GTask) task = G_TASK (user_data);
g_autoptr(GVariant) result_variant = NULL;
g_autoptr(GVariant) properties = NULL;
g_autoptr(GError) local_error = NULL;
g_autoptr(EpcAppFilter) app_filter = NULL;
gboolean is_whitelist;
g_auto(GStrv) app_list = NULL;
const gchar *content_rating_kind;
g_autoptr(GVariant) oars_variant = NULL;
g_autoptr(GHashTable) oars_map = NULL;
GetAppFilterData *data = g_task_get_task_data (task);
result_variant = g_dbus_connection_call_finish (connection, result, &local_error);
if (local_error != NULL)
{
g_autoptr(GError) app_filter_error = bus_error_to_app_filter_error (local_error,
data->user_id);
g_task_return_error (task, g_steal_pointer (&app_filter_error));
return;
}
/* Extract the properties we care about. They may be silently omitted from the
* results if we dont have permission to access them. */
properties = g_variant_get_child_value (result_variant, 0);
if (!g_variant_lookup (properties, "app-filter", "(b^as)",
&is_whitelist, &app_list))
{
g_task_return_new_error (task, EPC_APP_FILTER_ERROR,
EPC_APP_FILTER_ERROR_PERMISSION_DENIED,
_("Not allowed to query app filter data for user %u"),
data->user_id);
return;
}
if (!g_variant_lookup (properties, "oars-filter", "(&s@a{ss})",
&content_rating_kind, &oars_variant))
{
/* Default value. */
content_rating_kind = "oars-1.1";
oars_variant = g_variant_new ("@a{ss} {}");
}
/* Check that the OARS filter is in a format we support. Currently, thats
* only oars-1.0 and oars-1.1. */
if (!g_str_equal (content_rating_kind, "oars-1.0") &&
!g_str_equal (content_rating_kind, "oars-1.1"))
{
g_task_return_new_error (task, EPC_APP_FILTER_ERROR,
EPC_APP_FILTER_ERROR_INVALID_DATA,
_("OARS filter for user %u has an unrecognized kind %s"),
data->user_id, content_rating_kind);
return;
}
/* Success. Create an #EpcAppFilter object to contain the results. */
app_filter = g_new0 (EpcAppFilter, 1);
app_filter->ref_count = 1;
app_filter->user_id = data->user_id;
app_filter->app_list = g_steal_pointer (&app_list);
app_filter->app_list_type =
is_whitelist ? EPC_APP_FILTER_LIST_WHITELIST : EPC_APP_FILTER_LIST_BLACKLIST;
app_filter->oars_ratings = g_steal_pointer (&oars_variant);
g_task_return_pointer (task, g_steal_pointer (&app_filter),
(GDestroyNotify) epc_app_filter_unref);
} }
/** /**
@ -518,36 +525,121 @@ epc_get_app_filter_finish (GAsyncResult *result,
return g_task_propagate_pointer (G_TASK (result), error); return g_task_propagate_pointer (G_TASK (result), error);
} }
static void set_app_filter_get_bus_cb (GObject *obj, /**
GAsyncResult *result, * epc_set_app_filter:
gpointer user_data); * @connection: (nullable): a #GDBusConnection to the system bus, or %NULL to
static void set_app_filter (GDBusConnection *connection, * use the default
GTask *task); * @user_id: ID of the user to set the filter for, typically coming from getuid()
static void set_app_filter_cb (GObject *obj, * @app_filter: (transfer none): the app filter to set for the user
GAsyncResult *result, * @allow_interactive_authorization: %TRUE to allow interactive polkit
gpointer user_data); * authorization dialogues to be displayed during the call; %FALSE otherwise
static void set_oars_filter_cb (GObject *obj, * @cancellable: (nullable): a #GCancellable, or %NULL
GAsyncResult *result, * @error: return location for a #GError, or %NULL
gpointer user_data); *
static void set_app_filter_complete (GDBusConnection *connection, * Synchronous version of epc_set_app_filter_async().
GTask *task); *
* Returns: %TRUE on success, %FALSE otherwise
* Since: 0.1.0
*/
gboolean
epc_set_app_filter (GDBusConnection *connection,
uid_t user_id,
EpcAppFilter *app_filter,
gboolean allow_interactive_authorization,
GCancellable *cancellable,
GError **error)
{
g_autofree gchar *object_path = NULL;
g_autoptr(GVariant) app_filter_variant = NULL;
g_autoptr(GVariant) oars_filter_variant = NULL;
g_autoptr(GVariant) app_filter_result_variant = NULL;
g_autoptr(GVariant) oars_filter_result_variant = NULL;
g_autoptr(GError) local_error = NULL;
g_return_val_if_fail (connection == NULL || G_IS_DBUS_CONNECTION (connection), FALSE);
g_return_val_if_fail (app_filter != NULL, FALSE);
g_return_val_if_fail (app_filter->ref_count >= 1, FALSE);
g_return_val_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable), FALSE);
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
if (connection == NULL)
connection = g_bus_get_sync (G_BUS_TYPE_SYSTEM, cancellable, error);
if (connection == NULL)
return FALSE;
object_path = g_strdup_printf ("/org/freedesktop/Accounts/User%u", user_id);
app_filter_variant = _epc_app_filter_build_app_filter_variant (app_filter);
oars_filter_variant = g_variant_new ("(s@a{ss})", "oars-1.1",
app_filter->oars_ratings);
app_filter_result_variant =
g_dbus_connection_call_sync (connection,
"org.freedesktop.Accounts",
object_path,
"org.freedesktop.DBus.Properties",
"Set",
g_variant_new ("(ssv)",
"com.endlessm.ParentalControls.AppFilter",
"app-filter",
g_steal_pointer (&app_filter_variant)),
G_VARIANT_TYPE ("()"),
allow_interactive_authorization
? G_DBUS_CALL_FLAGS_ALLOW_INTERACTIVE_AUTHORIZATION
: G_DBUS_CALL_FLAGS_NONE,
-1, /* timeout, ms */
cancellable,
&local_error);
if (local_error != NULL)
{
g_propagate_error (error, bus_error_to_app_filter_error (local_error, user_id));
return FALSE;
}
oars_filter_result_variant =
g_dbus_connection_call_sync (connection,
"org.freedesktop.Accounts",
object_path,
"org.freedesktop.DBus.Properties",
"Set",
g_variant_new ("(ssv)",
"com.endlessm.ParentalControls.AppFilter",
"oars-filter",
g_steal_pointer (&oars_filter_variant)),
G_VARIANT_TYPE ("()"),
allow_interactive_authorization
? G_DBUS_CALL_FLAGS_ALLOW_INTERACTIVE_AUTHORIZATION
: G_DBUS_CALL_FLAGS_NONE,
-1, /* timeout, ms */
cancellable,
&local_error);
if (local_error != NULL)
{
g_propagate_error (error, bus_error_to_app_filter_error (local_error, user_id));
return FALSE;
}
return TRUE;
}
static void set_app_filter_thread_cb (GTask *task,
gpointer source_object,
gpointer task_data,
GCancellable *cancellable);
typedef struct typedef struct
{ {
GDBusConnection *connection; /* (nullable) (owned) */
uid_t user_id; uid_t user_id;
EpcAppFilter *app_filter; /* (owned) */ EpcAppFilter *app_filter; /* (owned) */
gboolean allow_interactive_authorization; gboolean allow_interactive_authorization;
GAsyncResult *set_app_filter_result; /* (nullable) (owned) */
GAsyncResult *set_oars_filter_result; /* (nullable) (owned) */
} SetAppFilterData; } SetAppFilterData;
static void static void
set_app_filter_data_free (SetAppFilterData *data) set_app_filter_data_free (SetAppFilterData *data)
{ {
g_clear_object (&data->connection);
epc_app_filter_unref (data->app_filter); epc_app_filter_unref (data->app_filter);
g_clear_object (&data->set_app_filter_result);
g_clear_object (&data->set_oars_filter_result);
g_free (data); g_free (data);
} }
@ -561,8 +653,8 @@ G_DEFINE_AUTOPTR_CLEANUP_FUNC (SetAppFilterData, set_app_filter_data_free)
* @app_filter: (transfer none): the app filter to set for the user * @app_filter: (transfer none): the app filter to set for the user
* @allow_interactive_authorization: %TRUE to allow interactive polkit * @allow_interactive_authorization: %TRUE to allow interactive polkit
* authorization dialogues to be displayed during the call; %FALSE otherwise * authorization dialogues to be displayed during the call; %FALSE otherwise
* @callback: a #GAsyncReadyCallback
* @cancellable: (nullable): a #GCancellable, or %NULL * @cancellable: (nullable): a #GCancellable, or %NULL
* @callback: a #GAsyncReadyCallback
* @user_data: user data to pass to @callback * @user_data: user data to pass to @callback
* *
* Asynchronously set the app filter settings for the given @user_id to the * Asynchronously set the app filter settings for the given @user_id to the
@ -586,7 +678,6 @@ epc_set_app_filter_async (GDBusConnection *connection,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data) gpointer user_data)
{ {
g_autoptr(GDBusConnection) connection_owned = NULL;
g_autoptr(GTask) task = NULL; g_autoptr(GTask) task = NULL;
g_autoptr(SetAppFilterData) data = NULL; g_autoptr(SetAppFilterData) data = NULL;
@ -605,148 +696,28 @@ epc_set_app_filter_async (GDBusConnection *connection,
g_task_set_task_data (task, g_steal_pointer (&data), g_task_set_task_data (task, g_steal_pointer (&data),
(GDestroyNotify) set_app_filter_data_free); (GDestroyNotify) set_app_filter_data_free);
if (connection == NULL) g_task_run_in_thread (task, set_app_filter_thread_cb);
g_bus_get (G_BUS_TYPE_SYSTEM, cancellable,
set_app_filter_get_bus_cb, g_steal_pointer (&task));
else
set_app_filter (connection, g_steal_pointer (&task));
} }
static void static void
set_app_filter_get_bus_cb (GObject *obj, set_app_filter_thread_cb (GTask *task,
GAsyncResult *result, gpointer source_object,
gpointer user_data) gpointer task_data,
GCancellable *cancellable)
{ {
g_autoptr(GTask) task = G_TASK (user_data); gboolean success;
g_autoptr(GDBusConnection) connection = NULL; SetAppFilterData *data = task_data;
g_autoptr(GError) local_error = NULL; g_autoptr(GError) local_error = NULL;
connection = g_bus_get_finish (result, &local_error); success = epc_set_app_filter (data->connection, data->user_id,
data->app_filter,
data->allow_interactive_authorization,
cancellable, &local_error);
if (local_error != NULL) if (local_error != NULL)
g_task_return_error (task, g_steal_pointer (&local_error)); g_task_return_error (task, g_steal_pointer (&local_error));
else else
set_app_filter (connection, g_steal_pointer (&task)); g_task_return_boolean (task, success);
}
static void
set_app_filter (GDBusConnection *connection,
GTask *task)
{
g_autofree gchar *object_path = NULL;
g_autoptr(GVariant) app_filter_variant = NULL;
g_autoptr(GVariant) oars_filter_variant = NULL;
GCancellable *cancellable;
SetAppFilterData *data = g_task_get_task_data (task);
cancellable = g_task_get_cancellable (task);
object_path = g_strdup_printf ("/org/freedesktop/Accounts/User%u",
data->user_id);
app_filter_variant = _epc_app_filter_build_app_filter_variant (data->app_filter);
oars_filter_variant = g_variant_new ("(s@a{ss})", "oars-1.1",
data->app_filter->oars_ratings);
g_dbus_connection_call (connection,
"org.freedesktop.Accounts",
object_path,
"org.freedesktop.DBus.Properties",
"Set",
g_variant_new ("(ssv)",
"com.endlessm.ParentalControls.AppFilter",
"app-filter",
g_steal_pointer (&app_filter_variant)),
G_VARIANT_TYPE ("()"),
data->allow_interactive_authorization
? G_DBUS_CALL_FLAGS_ALLOW_INTERACTIVE_AUTHORIZATION
: G_DBUS_CALL_FLAGS_NONE,
-1, /* timeout, ms */
cancellable,
set_app_filter_cb,
g_object_ref (task));
g_dbus_connection_call (connection,
"org.freedesktop.Accounts",
object_path,
"org.freedesktop.DBus.Properties",
"Set",
g_variant_new ("(ssv)",
"com.endlessm.ParentalControls.AppFilter",
"oars-filter",
g_steal_pointer (&oars_filter_variant)),
G_VARIANT_TYPE ("()"),
data->allow_interactive_authorization
? G_DBUS_CALL_FLAGS_ALLOW_INTERACTIVE_AUTHORIZATION
: G_DBUS_CALL_FLAGS_NONE,
-1, /* timeout, ms */
cancellable,
set_oars_filter_cb,
g_object_ref (task));
}
static void
set_app_filter_cb (GObject *obj,
GAsyncResult *result,
gpointer user_data)
{
GDBusConnection *connection = G_DBUS_CONNECTION (obj);
g_autoptr(GTask) task = G_TASK (user_data);
SetAppFilterData *data = g_task_get_task_data (task);
g_assert (data->set_app_filter_result == NULL);
data->set_app_filter_result = g_object_ref (result);
set_app_filter_complete (connection, task);
}
static void
set_oars_filter_cb (GObject *obj,
GAsyncResult *result,
gpointer user_data)
{
GDBusConnection *connection = G_DBUS_CONNECTION (obj);
g_autoptr(GTask) task = G_TASK (user_data);
SetAppFilterData *data = g_task_get_task_data (task);
g_assert (data->set_oars_filter_result == NULL);
data->set_oars_filter_result = g_object_ref (result);
set_app_filter_complete (connection, task);
}
static void
set_app_filter_complete (GDBusConnection *connection,
GTask *task)
{
g_autoptr(GVariant) app_filter_result_variant = NULL;
g_autoptr(GVariant) oars_filter_result_variant = NULL;
g_autoptr(GError) app_filter_error = NULL;
g_autoptr(GError) oars_filter_error = NULL;
SetAppFilterData *data = g_task_get_task_data (task);
if (data->set_app_filter_result == NULL ||
data->set_oars_filter_result == NULL)
return;
app_filter_result_variant = g_dbus_connection_call_finish (connection,
data->set_app_filter_result,
&app_filter_error);
oars_filter_result_variant = g_dbus_connection_call_finish (connection,
data->set_oars_filter_result,
&oars_filter_error);
if (app_filter_error != NULL)
{
g_task_return_error (task, bus_error_to_app_filter_error (app_filter_error,
data->user_id));
return;
}
if (oars_filter_error != NULL)
{
g_task_return_error (task, bus_error_to_app_filter_error (oars_filter_error,
data->user_id));
return;
}
g_task_return_boolean (task, TRUE);
} }
/** /**

View File

@ -106,6 +106,11 @@ gboolean epc_app_filter_is_flatpak_ref_allowed (EpcAppFilter *filter,
EpcAppFilterOarsValue epc_app_filter_get_oars_value (EpcAppFilter *filter, EpcAppFilterOarsValue epc_app_filter_get_oars_value (EpcAppFilter *filter,
const gchar *oars_section); const gchar *oars_section);
EpcAppFilter *epc_get_app_filter (GDBusConnection *connection,
uid_t user_id,
gboolean allow_interactive_authorization,
GCancellable *cancellable,
GError **error);
void epc_get_app_filter_async (GDBusConnection *connection, void epc_get_app_filter_async (GDBusConnection *connection,
uid_t user_id, uid_t user_id,
gboolean allow_interactive_authorization, gboolean allow_interactive_authorization,
@ -115,6 +120,12 @@ void epc_get_app_filter_async (GDBusConnection *connection,
EpcAppFilter *epc_get_app_filter_finish (GAsyncResult *result, EpcAppFilter *epc_get_app_filter_finish (GAsyncResult *result,
GError **error); GError **error);
gboolean epc_set_app_filter (GDBusConnection *connection,
uid_t user_id,
EpcAppFilter *app_filter,
gboolean allow_interactive_authorization,
GCancellable *cancellable,
GError **error);
void epc_set_app_filter_async (GDBusConnection *connection, void epc_set_app_filter_async (GDBusConnection *connection,
uid_t user_id, uid_t user_id,
EpcAppFilter *app_filter, EpcAppFilter *app_filter,