diff options
author | Guillaume Desmottes <guillaume.desmottes@collabora.co.uk> | 2011-11-02 21:48:23 +0800 |
---|---|---|
committer | Guillaume Desmottes <guillaume.desmottes@collabora.co.uk> | 2011-11-23 20:33:58 +0800 |
commit | 9389308505f9c90ba5a1ce0974ef7d870af7c7e6 (patch) | |
tree | 246d2a809fc15245350d5f1032bdda9a7b6073f8 | |
parent | 7f75f25ac7e93e198e23421298b619e6b92992da (diff) | |
download | gsoc2013-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.c | 60 | ||||
-rw-r--r-- | libempathy/empathy-auth-factory.h | 4 | ||||
-rw-r--r-- | src/empathy-auth-client.c | 36 |
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); |