aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGuillaume Desmottes <guillaume.desmottes@collabora.co.uk>2011-11-02 21:48:23 +0800
committerGuillaume Desmottes <guillaume.desmottes@collabora.co.uk>2011-11-23 20:33:58 +0800
commit9389308505f9c90ba5a1ce0974ef7d870af7c7e6 (patch)
tree246d2a809fc15245350d5f1032bdda9a7b6073f8
parent7f75f25ac7e93e198e23421298b619e6b92992da (diff)
downloadgsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.tar
gsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.tar.gz
gsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.tar.bz2
gsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.tar.lz
gsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.tar.xz
gsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.tar.zst
gsoc2013-empathy-9389308505f9c90ba5a1ce0974ef7d870af7c7e6.zip
Allow user to try another password if auth failed
https://bugzilla.gnome.org/show_bug.cgi?id=661640
-rw-r--r--libempathy/empathy-auth-factory.c60
-rw-r--r--libempathy/empathy-auth-factory.h4
-rw-r--r--src/empathy-auth-client.c36
3 files changed, 100 insertions, 0 deletions
diff --git a/libempathy/empathy-auth-factory.c b/libempathy/empathy-auth-factory.c
index 0aaea1c4a..fb178216d 100644
--- a/libempathy/empathy-auth-factory.c
+++ b/libempathy/empathy-auth-factory.c
@@ -53,6 +53,13 @@ struct _EmpathyAuthFactoryPriv {
EmpathyGoaAuthHandler *goa_handler;
#endif /* HAVE_GOA */
+ /* If an account failed to connect and user enters a new password to try, we
+ * store it in this hash table and will try to use it next time the account
+ * attemps to connect.
+ *
+ * reffed TpAccount -> owned password (gchar *) */
+ GHashTable *retry_passwords;
+
gboolean dispose_run;
};
@@ -184,6 +191,8 @@ server_sasl_handler_ready_cb (GObject *source,
else
{
TpChannel *channel;
+ const gchar *password;
+ TpAccount *account;
if (data->context != NULL)
tp_handle_channels_context_accept (data->context);
@@ -201,6 +210,29 @@ server_sasl_handler_ready_cb (GObject *source,
tp_g_signal_connect_object (handler, "auth-password-failed",
G_CALLBACK (sasl_handler_auth_password_failed_cb), data->self, 0);
+ /* Is there a retry password? */
+ account = empathy_server_sasl_handler_get_account (handler);
+
+ password = g_hash_table_lookup (data->self->priv->retry_passwords,
+ account);
+ if (password != NULL)
+ {
+ gboolean save;
+
+ DEBUG ("Use retry password");
+
+ /* We want to save this new password only if there is another
+ * (wrong) password saved. The SASL handler will only save it if it
+ * manages to connect. */
+ save = empathy_server_sasl_handler_has_password (handler);
+
+ empathy_server_sasl_handler_provide_password (handler,
+ password, save);
+
+ /* We only want to try this password once */
+ g_hash_table_remove (data->self->priv->retry_passwords, account);
+ }
+
g_signal_emit (data->self, signals[NEW_SERVER_SASL_HANDLER], 0,
handler);
}
@@ -493,6 +525,18 @@ observe_channels (TpBaseClient *client,
if (empathy_sasl_channel_supports_mechanism (data->channel,
"X-TELEPATHY-PASSWORD"))
{
+ if (g_hash_table_lookup (self->priv->retry_passwords, account) != NULL)
+ {
+ DEBUG ("We have a retry password for account %s, calling Claim",
+ tp_account_get_path_suffix (account));
+
+ tp_channel_dispatch_operation_claim_with_async (dispatch_operation,
+ client, password_claim_cb, data);
+
+ tp_observe_channels_context_accept (context);
+ return;
+ }
+
empathy_keyring_get_account_password_async (data->account,
get_password_cb, data);
tp_observe_channels_context_delay (context);
@@ -539,9 +583,13 @@ empathy_auth_factory_init (EmpathyAuthFactory *self)
self->priv->sasl_handlers = g_hash_table_new_full (g_str_hash, g_str_equal,
NULL, g_object_unref);
+
#ifdef HAVE_GOA
self->priv->goa_handler = empathy_goa_auth_handler_new ();
#endif /* HAVE_GOA */
+
+ self->priv->retry_passwords = g_hash_table_new_full (NULL, NULL,
+ g_object_unref, g_free);
}
static void
@@ -603,10 +651,13 @@ empathy_auth_factory_dispose (GObject *object)
priv->dispose_run = TRUE;
g_hash_table_unref (priv->sasl_handlers);
+
#ifdef HAVE_GOA
g_object_unref (priv->goa_handler);
#endif /* HAVE_GOA */
+ g_hash_table_unref (priv->retry_passwords);
+
G_OBJECT_CLASS (empathy_auth_factory_parent_class)->dispose (object);
}
@@ -668,3 +719,12 @@ empathy_auth_factory_register (EmpathyAuthFactory *self,
{
return tp_base_client_register (TP_BASE_CLIENT (self), error);
}
+
+void
+empathy_auth_factory_save_retry_password (EmpathyAuthFactory *self,
+ TpAccount *account,
+ const gchar *password)
+{
+ g_hash_table_insert (self->priv->retry_passwords,
+ g_object_ref (account), g_strdup (password));
+}
diff --git a/libempathy/empathy-auth-factory.h b/libempathy/empathy-auth-factory.h
index e84c13bcc..5d31b1edd 100644
--- a/libempathy/empathy-auth-factory.h
+++ b/libempathy/empathy-auth-factory.h
@@ -64,6 +64,10 @@ EmpathyAuthFactory * empathy_auth_factory_new (TpSimpleClientFactory *factory);
gboolean empathy_auth_factory_register (EmpathyAuthFactory *self,
GError **error);
+void empathy_auth_factory_save_retry_password (EmpathyAuthFactory *self,
+ TpAccount *account,
+ const gchar *password);
+
G_END_DECLS
#endif /* #ifndef __EMPATHY_AUTH_FACTORY_H__*/
diff --git a/src/empathy-auth-client.c b/src/empathy-auth-client.c
index bb7446847..5b3e48ce8 100644
--- a/src/empathy-auth-client.c
+++ b/src/empathy-auth-client.c
@@ -36,6 +36,7 @@
#include <libempathy/empathy-tls-verifier.h>
#include <libempathy/empathy-utils.h>
+#include <libempathy-gtk/empathy-bad-password-dialog.h>
#include <libempathy-gtk/empathy-password-dialog.h>
#include <libempathy-gtk/empathy-tls-dialog.h>
#include <libempathy-gtk/empathy-ui-utils.h>
@@ -235,6 +236,38 @@ auth_factory_new_sasl_handler_cb (EmpathyAuthFactory *factory,
}
}
+static void
+retry_account_cb (GtkWidget *dialog,
+ TpAccount *account,
+ const gchar *password,
+ EmpathyAuthFactory *factory)
+{
+ DEBUG ("Try reconnecting to %s", tp_account_get_path_suffix (account));
+
+ empathy_auth_factory_save_retry_password (factory, account, password);
+
+ tp_account_reconnect_async (account, NULL, NULL);
+}
+
+static void
+auth_factory_auth_passsword_failed (EmpathyAuthFactory *factory,
+ TpAccount *account,
+ const gchar *password,
+ gpointer user_data)
+{
+ GtkWidget *dialog;
+
+ DEBUG ("Authentification on %s failed, popup password dialog",
+ tp_account_get_path_suffix (account));
+
+ dialog = empathy_bad_password_dialog_new (account, password);
+
+ tp_g_signal_connect_object (dialog, "retry",
+ G_CALLBACK (retry_account_cb), factory, 0);
+
+ gtk_widget_show (dialog);
+}
+
int
main (int argc,
char **argv)
@@ -294,6 +327,9 @@ main (int argc,
g_signal_connect (factory, "new-server-sasl-handler",
G_CALLBACK (auth_factory_new_sasl_handler_cb), NULL);
+ g_signal_connect (factory, "auth-password-failed",
+ G_CALLBACK (auth_factory_auth_passsword_failed), NULL);
+
if (!empathy_auth_factory_register (factory, &error))
{
g_critical ("Failed to register the auth factory: %s\n", error->message);