mirror of
https://gitlab.freedesktop.org/libfprint/fprintd.git
synced 2026-02-04 09:30:26 +01:00
Fprintd is dependent on the deprecated dbus-glib, also this doesn't provide various features we can take advantage of, like the ones for async authentication mechanism. So, remove all the dbus-glib dependencies and simplify the code, but without any further refactor, and keeping everything as it used to work, while this will give room for further improvements in subsequent commits. Internally, we just use dbus-codegen to generate the skeletons, and we use the generated FprintdDBusManager with composition, while we implement the device skeleton interface in FprintDevice, so that we don't have to use it as a proxy, and keep being closer to what it used to be with dbus-glib. Fixes: #61
243 lines
6.5 KiB
C
243 lines
6.5 KiB
C
/*
|
|
* fprintd example to verify a fingerprint
|
|
* Copyright (C) 2008 Daniel Drake <dsd@gentoo.org>
|
|
* Copyright (C) 2020 Marco Trevisan <marco.trevisan@canonical.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <locale.h>
|
|
#include <gio/gio.h>
|
|
#include "fprintd-dbus.h"
|
|
|
|
static FprintDBusManager *manager = NULL;
|
|
static GDBusConnection *connection = NULL;
|
|
static char *finger_name = NULL;
|
|
static gboolean g_fatal_warnings = FALSE;
|
|
static char **usernames = NULL;
|
|
|
|
static void create_manager(void)
|
|
{
|
|
g_autoptr(GError) error = NULL;
|
|
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SYSTEM, NULL, &error);
|
|
if (connection == NULL) {
|
|
g_print("Failed to connect to session bus: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
manager = fprint_dbus_manager_proxy_new_sync (connection,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
"net.reactivated.Fprint",
|
|
"/net/reactivated/Fprint/Manager",
|
|
NULL, &error);
|
|
if (manager == NULL) {
|
|
g_print ("Failed to get Fprintd manager: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
}
|
|
|
|
static FprintDBusDevice *open_device (const char *username)
|
|
{
|
|
g_autoptr(FprintDBusDevice) dev = NULL;
|
|
GError *error = NULL;
|
|
gchar *path;
|
|
|
|
if (!fprint_dbus_manager_call_get_default_device_sync (manager, &path,
|
|
NULL, &error)) {
|
|
g_print("Impossible to verify: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
g_print("Using device %s\n", path);
|
|
|
|
dev = fprint_dbus_device_proxy_new_sync (connection,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
"net.reactivated.Fprint",
|
|
path, NULL, &error);
|
|
|
|
g_free (path);
|
|
|
|
if (error) {
|
|
g_print ("failed to connect to device: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
if (!fprint_dbus_device_call_claim_sync (dev, username, NULL, &error)) {
|
|
g_print("failed to claim device: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
return g_steal_pointer (&dev);
|
|
}
|
|
|
|
static void find_finger (FprintDBusDevice *dev, const char *username)
|
|
{
|
|
GError *error = NULL;
|
|
char **fingers;
|
|
guint i;
|
|
|
|
if (!fprint_dbus_device_call_list_enrolled_fingers_sync (dev, username,
|
|
&fingers,
|
|
NULL, &error)) {
|
|
g_print("ListEnrolledFingers failed: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
if (fingers == NULL || g_strv_length (fingers) == 0) {
|
|
g_print("No fingers enrolled for this device.\n");
|
|
exit(1);
|
|
}
|
|
|
|
g_print("Listing enrolled fingers:\n");
|
|
for (i = 0; fingers[i] != NULL; i++) {
|
|
g_print(" - #%d: %s\n", i, fingers[i]);
|
|
}
|
|
|
|
if (finger_name && !g_str_equal (finger_name, "any") &&
|
|
!g_strv_contains ((const char **) fingers, finger_name)) {
|
|
g_print("Finger '%s' not enrolled for user %s.\n", finger_name,
|
|
username);
|
|
g_free (finger_name);
|
|
exit(1);
|
|
}
|
|
|
|
if (finger_name == NULL) {
|
|
finger_name = g_strdup (fingers[0]);
|
|
}
|
|
|
|
g_strfreev (fingers);
|
|
}
|
|
|
|
static void verify_result(GObject *object, const char *result, gboolean done, void *user_data)
|
|
{
|
|
gboolean *verify_completed = user_data;
|
|
g_print("Verify result: %s (%s)\n", result, done ? "done" : "not done");
|
|
if (done != FALSE)
|
|
*verify_completed = TRUE;
|
|
}
|
|
|
|
static void verify_finger_selected(GObject *object, const char *name, void *user_data)
|
|
{
|
|
g_print("Verifying: %s\n", name);
|
|
}
|
|
|
|
static void proxy_signal_cb (GDBusProxy *proxy,
|
|
const gchar *sender_name,
|
|
const gchar *signal_name,
|
|
GVariant *parameters,
|
|
gpointer user_data)
|
|
{
|
|
if (g_str_equal (signal_name, "VerifyStatus")) {
|
|
const gchar *result;
|
|
gboolean done;
|
|
|
|
g_variant_get (parameters, "(&sb)", &result, &done);
|
|
verify_result (G_OBJECT (proxy), result, done, user_data);
|
|
} else if (g_str_equal (signal_name, "VerifyFingerSelected")) {
|
|
const gchar *name;
|
|
|
|
g_variant_get (parameters, "(&s)", &name);
|
|
verify_finger_selected (G_OBJECT (proxy), name, user_data);
|
|
}
|
|
}
|
|
|
|
static void do_verify (FprintDBusDevice *dev)
|
|
{
|
|
GError *error = NULL;
|
|
gboolean verify_completed = FALSE;
|
|
|
|
g_signal_connect (dev, "g-signal", G_CALLBACK (proxy_signal_cb),
|
|
&verify_completed);
|
|
|
|
if (!fprint_dbus_device_call_verify_start_sync (dev, finger_name, NULL,
|
|
&error)) {
|
|
g_print("VerifyStart failed: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
while (!verify_completed)
|
|
g_main_context_iteration(NULL, TRUE);
|
|
|
|
|
|
g_signal_handlers_disconnect_by_func (dev, proxy_signal_cb,
|
|
&verify_completed);
|
|
|
|
if (!fprint_dbus_device_call_verify_stop_sync (dev, NULL, &error)) {
|
|
g_print("VerifyStop failed: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
}
|
|
|
|
static void release_device (FprintDBusDevice *dev)
|
|
{
|
|
GError *error = NULL;
|
|
if (!fprint_dbus_device_call_release_sync (dev, NULL, &error)) {
|
|
g_print("ReleaseDevice failed: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
}
|
|
|
|
static const GOptionEntry entries[] = {
|
|
{ "finger", 'f', 0, G_OPTION_ARG_STRING, &finger_name, "Finger selected to verify (default is automatic)", NULL },
|
|
{"g-fatal-warnings", 0, 0, G_OPTION_ARG_NONE, &g_fatal_warnings, "Make all warnings fatal", NULL},
|
|
{ G_OPTION_REMAINING, '\0', 0, G_OPTION_ARG_STRING_ARRAY, &usernames, NULL, "[username]" },
|
|
{ NULL }
|
|
};
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
g_autoptr(FprintDBusDevice) dev = NULL;
|
|
GOptionContext *context;
|
|
GError *err = NULL;
|
|
const char *username = NULL;
|
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
context = g_option_context_new ("Verify a fingerprint");
|
|
g_option_context_add_main_entries (context, entries, NULL);
|
|
|
|
if (g_option_context_parse (context, &argc, &argv, &err) == FALSE) {
|
|
g_print ("couldn't parse command-line options: %s\n", err->message);
|
|
g_error_free (err);
|
|
return 1;
|
|
}
|
|
|
|
if (usernames == NULL) {
|
|
username = "";
|
|
} else {
|
|
username = usernames[0];
|
|
}
|
|
|
|
if (g_fatal_warnings) {
|
|
GLogLevelFlags fatal_mask;
|
|
|
|
fatal_mask = g_log_set_always_fatal (G_LOG_FATAL_MASK);
|
|
fatal_mask |= G_LOG_LEVEL_WARNING | G_LOG_LEVEL_CRITICAL;
|
|
g_log_set_always_fatal (fatal_mask);
|
|
}
|
|
|
|
create_manager();
|
|
|
|
dev = open_device(username);
|
|
find_finger(dev, username);
|
|
do_verify(dev);
|
|
release_device(dev);
|
|
return 0;
|
|
}
|
|
|