/* GIO - GLib Input, Output and Streaming Library
*
* Copyright (C) 2011 Collabora, Ltd.
*
* 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 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 .
*
* Author: Stef Walter
*/
#include "config.h"
#include
#include
#include
#ifdef G_OS_WIN32
#include
#endif
#include "gtlsconsoleinteraction.h"
/*
* WARNING: This is not the example you're looking for [slow hand wave]. This
* is not industrial strength, it's just for testing. It uses embarassing
* functions like getpass() and does lazy things with threads.
*/
G_DEFINE_TYPE (GTlsConsoleInteraction, g_tls_console_interaction, G_TYPE_TLS_INTERACTION);
#if defined(G_OS_WIN32) || defined(__BIONIC__)
/* win32 doesn't have getpass() */
#include
#ifndef BUFSIZ
#define BUFSIZ 8192
#endif
static gchar *
getpass (const gchar *prompt)
{
static gchar buf[BUFSIZ];
gint i;
g_printf ("%s", prompt);
fflush (stdout);
for (i = 0; i < BUFSIZ - 1; ++i)
{
#ifdef __BIONIC__
buf[i] = getc (stdin);
#else
buf[i] = _getch ();
#endif
if (buf[i] == '\r')
break;
}
buf[i] = '\0';
g_printf ("\n");
return &buf[0];
}
#endif
static GTlsInteractionResult
g_tls_console_interaction_ask_password (GTlsInteraction *interaction,
GTlsPassword *password,
GCancellable *cancellable,
GError **error)
{
const gchar *value;
gchar *prompt;
prompt = g_strdup_printf ("Password \"%s\"': ", g_tls_password_get_description (password));
value = getpass (prompt);
g_free (prompt);
if (g_cancellable_set_error_if_cancelled (cancellable, error))
return G_TLS_INTERACTION_FAILED;
g_tls_password_set_value (password, (guchar *)value, -1);
return G_TLS_INTERACTION_HANDLED;
}
static void
ask_password_with_getpass (GTask *task,
gpointer object,
gpointer task_data,
GCancellable *cancellable)
{
GTlsPassword *password = task_data;
GError *error = NULL;
g_tls_console_interaction_ask_password (G_TLS_INTERACTION (object), password,
cancellable, &error);
if (error != NULL)
g_task_return_error (task, error);
else
g_task_return_int (task, G_TLS_INTERACTION_HANDLED);
}
static void
g_tls_console_interaction_ask_password_async (GTlsInteraction *interaction,
GTlsPassword *password,
GCancellable *cancellable,
GAsyncReadyCallback callback,
gpointer user_data)
{
GTask *task;
task = g_task_new (interaction, cancellable, callback, user_data);
g_task_set_task_data (task, g_object_ref (password), g_object_unref);
g_task_run_in_thread (task, ask_password_with_getpass);
g_object_unref (task);
}
static GTlsInteractionResult
g_tls_console_interaction_ask_password_finish (GTlsInteraction *interaction,
GAsyncResult *result,
GError **error)
{
GTlsInteractionResult ret;
g_return_val_if_fail (g_task_is_valid (result, interaction),
G_TLS_INTERACTION_FAILED);
ret = g_task_propagate_int (G_TASK (result), error);
if (ret == (GTlsInteractionResult)-1)
return G_TLS_INTERACTION_FAILED;
else
return ret;
}
static void
g_tls_console_interaction_init (GTlsConsoleInteraction *interaction)
{
}
static void
g_tls_console_interaction_class_init (GTlsConsoleInteractionClass *klass)
{
GTlsInteractionClass *interaction_class = G_TLS_INTERACTION_CLASS (klass);
interaction_class->ask_password = g_tls_console_interaction_ask_password;
interaction_class->ask_password_async = g_tls_console_interaction_ask_password_async;
interaction_class->ask_password_finish = g_tls_console_interaction_ask_password_finish;
}
GTlsInteraction *
g_tls_console_interaction_new (void)
{
return g_object_new (G_TYPE_TLS_CONSOLE_INTERACTION, NULL);
}