rename accountopt.[ch] to purpleaccountoption.[ch]
[pidgin-git.git] / libpurple / accounts.c
blob4e387688bf5e23e25833254b56221beeaa20aa76
1 /* purple
3 * Purple is the legal property of its developers, whose names are too numerous
4 * to list here. Please refer to the COPYRIGHT file distributed with this
5 * source distribution.
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
21 #include "internal.h"
22 #include "accounts.h"
23 #include "core.h"
24 #include "debug.h"
25 #include "enums.h"
26 #include "network.h"
27 #include "pounce.h"
29 static PurpleAccountUiOps *account_ui_ops = NULL;
31 static GList *accounts = NULL;
32 static guint save_timer = 0;
33 static gboolean accounts_loaded = FALSE;
35 /*********************************************************************
36 * Writing to disk *
37 *********************************************************************/
38 static PurpleXmlNode *
39 accounts_to_xmlnode(void)
41 PurpleXmlNode *node, *child;
42 GList *cur;
44 node = purple_xmlnode_new("account");
45 purple_xmlnode_set_attrib(node, "version", "1.0");
47 for (cur = purple_accounts_get_all(); cur != NULL; cur = cur->next)
49 child = _purple_account_to_xmlnode(cur->data);
50 purple_xmlnode_insert_child(node, child);
53 return node;
56 static void
57 sync_accounts(void)
59 PurpleXmlNode *node;
60 char *data;
62 if (!accounts_loaded)
64 purple_debug_error("accounts", "Attempted to save accounts before "
65 "they were read!\n");
66 return;
69 node = accounts_to_xmlnode();
70 data = purple_xmlnode_to_formatted_str(node, NULL);
71 purple_util_write_data_to_config_file("accounts.xml", data, -1);
72 g_free(data);
73 purple_xmlnode_free(node);
76 static gboolean
77 save_cb(gpointer data)
79 sync_accounts();
80 save_timer = 0;
81 return FALSE;
84 void
85 purple_accounts_schedule_save(void)
87 if (save_timer == 0)
88 save_timer = g_timeout_add_seconds(5, save_cb, NULL);
91 static void
92 migrate_icq_server(PurpleAccount *account)
94 /* Migrate the login server setting for ICQ accounts. See
95 * 'mtn log --last 1 --no-graph --from b6d7712e90b68610df3bd2d8cbaf46d94c8b3794'
96 * for details on the change. */
98 if(purple_strequal(purple_account_get_protocol_id(account), "prpl-icq")) {
99 const char *tmp = purple_account_get_string(account, "server", NULL);
101 /* Non-secure server */
102 if(purple_strequal(tmp, "login.messaging.aol.com") ||
103 purple_strequal(tmp, "login.oscar.aol.com"))
104 purple_account_set_string(account, "server", "login.icq.com");
106 /* Secure server */
107 if(purple_strequal(tmp, "slogin.oscar.aol.com"))
108 purple_account_set_string(account, "server", "slogin.icq.com");
112 static void
113 migrate_xmpp_encryption(PurpleAccount *account)
115 /* When this is removed, nuke the "old_ssl" and "require_tls" settings */
116 if (g_str_equal(purple_account_get_protocol_id(account), "prpl-jabber")) {
117 const char *sec = purple_account_get_string(account, "connection_security", "");
119 if (g_str_equal("", sec)) {
120 const char *val = "require_tls";
121 if (purple_account_get_bool(account, "old_ssl", FALSE))
122 val = "old_ssl";
123 else if (!purple_account_get_bool(account, "require_tls", TRUE))
124 val = "opportunistic_tls";
126 purple_account_set_string(account, "connection_security", val);
131 static void
132 parse_settings(PurpleXmlNode *node, PurpleAccount *account)
134 const char *ui;
135 PurpleXmlNode *child;
137 /* Get the UI string, if these are UI settings */
138 ui = purple_xmlnode_get_attrib(node, "ui");
140 /* Read settings, one by one */
141 for (child = purple_xmlnode_get_child(node, "setting"); child != NULL;
142 child = purple_xmlnode_get_next_twin(child))
144 const char *name, *str_type;
145 PurplePrefType type;
146 char *data;
148 name = purple_xmlnode_get_attrib(child, "name");
149 if (name == NULL)
150 /* Ignore this setting */
151 continue;
153 str_type = purple_xmlnode_get_attrib(child, "type");
154 if (str_type == NULL)
155 /* Ignore this setting */
156 continue;
158 if (purple_strequal(str_type, "string"))
159 type = PURPLE_PREF_STRING;
160 else if (purple_strequal(str_type, "int"))
161 type = PURPLE_PREF_INT;
162 else if (purple_strequal(str_type, "bool"))
163 type = PURPLE_PREF_BOOLEAN;
164 else
165 /* Ignore this setting */
166 continue;
168 data = purple_xmlnode_get_data(child);
169 if (data == NULL)
170 /* Ignore this setting */
171 continue;
173 if (ui == NULL)
175 if (type == PURPLE_PREF_STRING)
176 purple_account_set_string(account, name, data);
177 else if (type == PURPLE_PREF_INT)
178 purple_account_set_int(account, name, atoi(data));
179 else if (type == PURPLE_PREF_BOOLEAN)
180 purple_account_set_bool(account, name,
181 (*data == '0' ? FALSE : TRUE));
182 } else {
183 if (type == PURPLE_PREF_STRING)
184 purple_account_set_ui_string(account, ui, name, data);
185 else if (type == PURPLE_PREF_INT)
186 purple_account_set_ui_int(account, ui, name, atoi(data));
187 else if (type == PURPLE_PREF_BOOLEAN)
188 purple_account_set_ui_bool(account, ui, name,
189 (*data == '0' ? FALSE : TRUE));
192 g_free(data);
195 /* we do this here because we need access to account settings to determine
196 * if we can/should migrate an ICQ account's server setting */
197 migrate_icq_server(account);
198 /* we do this here because we need to do it before the user views the
199 * Edit Account dialog. */
200 migrate_xmpp_encryption(account);
203 static GList *
204 parse_status_attrs(PurpleXmlNode *node, PurpleStatus *status)
206 GList *list = NULL;
207 PurpleXmlNode *child;
208 GValue *attr_value;
210 for (child = purple_xmlnode_get_child(node, "attribute"); child != NULL;
211 child = purple_xmlnode_get_next_twin(child))
213 const char *id = purple_xmlnode_get_attrib(child, "id");
214 const char *value = purple_xmlnode_get_attrib(child, "value");
216 if (!id || !*id || !value || !*value)
217 continue;
219 attr_value = purple_status_get_attr_value(status, id);
220 if (!attr_value)
221 continue;
223 list = g_list_append(list, (char *)id);
225 switch (G_VALUE_TYPE(attr_value))
227 case G_TYPE_STRING:
228 list = g_list_append(list, (char *)value);
229 break;
230 case G_TYPE_INT:
231 case G_TYPE_BOOLEAN:
233 int v;
234 if (sscanf(value, "%d", &v) == 1)
235 list = g_list_append(list, GINT_TO_POINTER(v));
236 else
237 list = g_list_remove(list, id);
238 break;
240 default:
241 break;
245 return list;
248 static void
249 parse_status(PurpleXmlNode *node, PurpleAccount *account)
251 gboolean active = FALSE;
252 const char *data;
253 const char *type;
254 PurpleXmlNode *child;
255 GList *attrs = NULL;
257 /* Get the active/inactive state */
258 data = purple_xmlnode_get_attrib(node, "active");
259 if (data == NULL)
260 return;
261 if (g_ascii_strcasecmp(data, "true") == 0)
262 active = TRUE;
263 else if (g_ascii_strcasecmp(data, "false") == 0)
264 active = FALSE;
265 else
266 return;
268 /* Get the type of the status */
269 type = purple_xmlnode_get_attrib(node, "type");
270 if (type == NULL)
271 return;
273 /* Read attributes into a GList */
274 child = purple_xmlnode_get_child(node, "attributes");
275 if (child != NULL)
277 attrs = parse_status_attrs(child,
278 purple_account_get_status(account, type));
281 purple_account_set_status_list(account, type, active, attrs);
283 g_list_free(attrs);
286 static void
287 parse_statuses(PurpleXmlNode *node, PurpleAccount *account)
289 PurpleXmlNode *child;
291 for (child = purple_xmlnode_get_child(node, "status"); child != NULL;
292 child = purple_xmlnode_get_next_twin(child))
294 parse_status(child, account);
298 static void
299 parse_proxy_info(PurpleXmlNode *node, PurpleAccount *account)
301 PurpleProxyInfo *proxy_info;
302 PurpleXmlNode *child;
303 char *data;
305 proxy_info = purple_proxy_info_new();
307 /* Use the global proxy settings, by default */
308 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_USE_GLOBAL);
310 /* Read proxy type */
311 child = purple_xmlnode_get_child(node, "type");
312 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
314 if (purple_strequal(data, "global"))
315 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_USE_GLOBAL);
316 else if (purple_strequal(data, "none"))
317 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_NONE);
318 else if (purple_strequal(data, "http"))
319 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_HTTP);
320 else if (purple_strequal(data, "socks4"))
321 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_SOCKS4);
322 else if (purple_strequal(data, "socks5"))
323 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_SOCKS5);
324 else if (purple_strequal(data, "tor"))
325 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_TOR);
326 else if (purple_strequal(data, "envvar"))
327 purple_proxy_info_set_proxy_type(proxy_info, PURPLE_PROXY_USE_ENVVAR);
328 else
330 purple_debug_error("accounts", "Invalid proxy type found when "
331 "loading account information for %s\n",
332 purple_account_get_username(account));
334 g_free(data);
337 /* Read proxy host */
338 child = purple_xmlnode_get_child(node, "host");
339 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
341 purple_proxy_info_set_host(proxy_info, data);
342 g_free(data);
345 /* Read proxy port */
346 child = purple_xmlnode_get_child(node, "port");
347 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
349 purple_proxy_info_set_port(proxy_info, atoi(data));
350 g_free(data);
353 /* Read proxy username */
354 child = purple_xmlnode_get_child(node, "username");
355 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
357 purple_proxy_info_set_username(proxy_info, data);
358 g_free(data);
361 /* Read proxy password */
362 child = purple_xmlnode_get_child(node, "password");
363 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
365 purple_proxy_info_set_password(proxy_info, data);
366 g_free(data);
369 /* If there are no values set then proxy_info NULL */
370 if ((purple_proxy_info_get_proxy_type(proxy_info) == PURPLE_PROXY_USE_GLOBAL) &&
371 (purple_proxy_info_get_host(proxy_info) == NULL) &&
372 (purple_proxy_info_get_port(proxy_info) == 0) &&
373 (purple_proxy_info_get_username(proxy_info) == NULL) &&
374 (purple_proxy_info_get_password(proxy_info) == NULL))
376 purple_proxy_info_destroy(proxy_info);
377 return;
380 purple_account_set_proxy_info(account, proxy_info);
383 static void
384 parse_current_error(PurpleXmlNode *node, PurpleAccount *account)
386 guint type;
387 char *type_str = NULL, *description = NULL;
388 PurpleXmlNode *child;
389 PurpleConnectionErrorInfo *current_error = NULL;
391 child = purple_xmlnode_get_child(node, "type");
392 if (child == NULL || (type_str = purple_xmlnode_get_data(child)) == NULL)
393 return;
394 type = atoi(type_str);
395 g_free(type_str);
397 if (type > PURPLE_CONNECTION_ERROR_OTHER_ERROR)
399 purple_debug_error("accounts",
400 "Invalid PurpleConnectionError value %d found when "
401 "loading account information for %s\n",
402 type, purple_account_get_username(account));
403 type = PURPLE_CONNECTION_ERROR_OTHER_ERROR;
406 child = purple_xmlnode_get_child(node, "description");
407 if (child)
408 description = purple_xmlnode_get_data(child);
409 if (description == NULL)
410 description = g_strdup("");
412 current_error = g_new0(PurpleConnectionErrorInfo, 1);
413 current_error->type = type;
414 current_error->description = description;
416 _purple_account_set_current_error(account, current_error);
419 static PurpleAccount *
420 parse_account(PurpleXmlNode *node)
422 PurpleAccount *ret;
423 PurpleXmlNode *child;
424 char *protocol_id = NULL;
425 char *name = NULL;
426 char *data;
428 child = purple_xmlnode_get_child(node, "protocol");
429 if (child != NULL)
430 protocol_id = purple_xmlnode_get_data(child);
432 child = purple_xmlnode_get_child(node, "name");
433 if (child != NULL)
434 name = purple_xmlnode_get_data(child);
435 if (name == NULL)
437 /* Do we really need to do this? */
438 child = purple_xmlnode_get_child(node, "username");
439 if (child != NULL)
440 name = purple_xmlnode_get_data(child);
443 if ((protocol_id == NULL) || (name == NULL))
445 g_free(protocol_id);
446 g_free(name);
447 return NULL;
450 ret = purple_account_new(name, protocol_id);
451 g_free(name);
452 g_free(protocol_id);
454 /* Read the alias */
455 child = purple_xmlnode_get_child(node, "alias");
456 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
458 if (*data != '\0')
459 purple_account_set_private_alias(ret, data);
460 g_free(data);
463 /* Read the statuses */
464 child = purple_xmlnode_get_child(node, "statuses");
465 if (child != NULL)
467 parse_statuses(child, ret);
470 /* Read the userinfo */
471 child = purple_xmlnode_get_child(node, "userinfo");
472 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
474 purple_account_set_user_info(ret, data);
475 g_free(data);
478 /* Read an old buddyicon */
479 child = purple_xmlnode_get_child(node, "buddyicon");
480 if ((child != NULL) && ((data = purple_xmlnode_get_data(child)) != NULL))
482 const char *dirname = purple_buddy_icons_get_cache_dir();
483 char *filename = g_build_filename(dirname, data, NULL);
484 gchar *contents;
485 gsize len;
487 if (g_file_get_contents(filename, &contents, &len, NULL))
489 purple_buddy_icons_set_account_icon(ret, (guchar *)contents, len);
492 g_free(filename);
493 g_free(data);
496 /* Read settings (both core and UI) */
497 for (child = purple_xmlnode_get_child(node, "settings"); child != NULL;
498 child = purple_xmlnode_get_next_twin(child))
500 parse_settings(child, ret);
503 /* Read proxy */
504 child = purple_xmlnode_get_child(node, "proxy");
505 if (child != NULL)
507 parse_proxy_info(child, ret);
510 /* Read current error */
511 child = purple_xmlnode_get_child(node, "current_error");
512 if (child != NULL)
514 parse_current_error(child, ret);
517 /* Read the password */
518 child = purple_xmlnode_get_child(node, "password");
519 if (child != NULL)
521 const char *keyring_id = purple_xmlnode_get_attrib(child, "keyring_id");
522 const char *mode = purple_xmlnode_get_attrib(child, "mode");
523 gboolean result;
525 data = purple_xmlnode_get_data(child);
526 result = purple_keyring_import_password(ret, keyring_id, mode, data, NULL);
528 if (result == TRUE || purple_keyring_get_inuse() == NULL) {
529 purple_account_set_remember_password(ret, TRUE);
530 } else {
531 purple_debug_error("accounts", "Failed to import password.\n");
533 purple_str_wipe(data);
536 return ret;
539 static void
540 load_accounts(void)
542 PurpleXmlNode *node, *child;
544 accounts_loaded = TRUE;
546 node = purple_util_read_xml_from_config_file("accounts.xml", _("accounts"));
548 if (node == NULL)
549 return;
551 for (child = purple_xmlnode_get_child(node, "account"); child != NULL;
552 child = purple_xmlnode_get_next_twin(child))
554 PurpleAccount *new_acct;
555 new_acct = parse_account(child);
556 purple_accounts_add(new_acct);
559 purple_xmlnode_free(node);
561 _purple_buddy_icons_account_loaded_cb();
564 void
565 purple_accounts_add(PurpleAccount *account)
567 g_return_if_fail(account != NULL);
569 if (g_list_find(accounts, account) != NULL)
570 return;
572 accounts = g_list_append(accounts, account);
574 purple_accounts_schedule_save();
576 purple_signal_emit(purple_accounts_get_handle(), "account-added", account);
579 void
580 purple_accounts_remove(PurpleAccount *account)
582 g_return_if_fail(account != NULL);
584 accounts = g_list_remove(accounts, account);
586 purple_accounts_schedule_save();
588 /* Clearing the error ensures that account-error-changed is emitted,
589 * which is the end of the guarantee that the the error's pointer is
590 * valid.
592 purple_account_clear_current_error(account);
593 purple_signal_emit(purple_accounts_get_handle(), "account-removed", account);
596 static void
597 purple_accounts_delete_set(PurpleAccount *account, GError *error, gpointer data)
599 g_object_unref(G_OBJECT(account));
602 void
603 purple_accounts_delete(PurpleAccount *account)
605 PurpleBlistNode *gnode, *cnode, *bnode;
606 GList *iter;
608 g_return_if_fail(account != NULL);
611 * Disable the account before blowing it out of the water.
612 * Conceptually it probably makes more sense to disable the
613 * account for all UIs rather than the just the current UI,
614 * but it doesn't really matter.
616 purple_account_set_enabled(account, purple_core_get_ui(), FALSE);
618 purple_notify_close_with_handle(account);
619 purple_request_close_with_handle(account);
621 purple_accounts_remove(account);
623 /* Remove this account's buddies */
624 for (gnode = purple_blist_get_default_root(); gnode != NULL;
625 gnode = purple_blist_node_get_sibling_next(gnode)) {
626 if (!PURPLE_IS_GROUP(gnode))
627 continue;
629 cnode = purple_blist_node_get_first_child(gnode);
630 while (cnode) {
631 PurpleBlistNode *cnode_next = purple_blist_node_get_sibling_next(cnode);
633 if(PURPLE_IS_CONTACT(cnode)) {
634 bnode = purple_blist_node_get_first_child(cnode);
635 while (bnode) {
636 PurpleBlistNode *bnode_next = purple_blist_node_get_sibling_next(bnode);
638 if (PURPLE_IS_BUDDY(bnode)) {
639 PurpleBuddy *b = (PurpleBuddy *)bnode;
641 if (purple_buddy_get_account(b) == account)
642 purple_blist_remove_buddy(b);
644 bnode = bnode_next;
646 } else if (PURPLE_IS_CHAT(cnode)) {
647 PurpleChat *c = (PurpleChat *)cnode;
649 if (purple_chat_get_account(c) == account)
650 purple_blist_remove_chat(c);
652 cnode = cnode_next;
656 /* Remove any open conversation for this account */
657 for (iter = purple_conversations_get_all(); iter; ) {
658 PurpleConversation *conv = iter->data;
659 iter = iter->next;
660 if (purple_conversation_get_account(conv) == account)
661 g_object_unref(conv);
664 /* Remove this account's pounces */
665 purple_pounce_destroy_all_by_account(account);
667 /* This will cause the deletion of an old buddy icon. */
668 purple_buddy_icons_set_account_icon(account, NULL, 0);
670 /* This is async because we do not want the
671 * account being overwritten before we are done.
673 purple_keyring_set_password(account, NULL,
674 purple_accounts_delete_set, NULL);
677 void
678 purple_accounts_reorder(PurpleAccount *account, guint new_index)
680 gint index;
681 GList *l;
683 g_return_if_fail(account != NULL);
684 g_return_if_fail(new_index <= g_list_length(accounts));
686 index = g_list_index(accounts, account);
688 if (index < 0) {
689 purple_debug_error("accounts",
690 "Unregistered account (%s) discovered during reorder!\n",
691 purple_account_get_username(account));
692 return;
695 l = g_list_nth(accounts, index);
697 if (new_index > (guint)index)
698 new_index--;
700 /* Remove the old one. */
701 accounts = g_list_delete_link(accounts, l);
703 /* Insert it where it should go. */
704 accounts = g_list_insert(accounts, account, new_index);
706 purple_accounts_schedule_save();
709 GList *
710 purple_accounts_get_all(void)
712 return accounts;
715 GList *
716 purple_accounts_get_all_active(void)
718 GList *list = NULL;
719 GList *all = purple_accounts_get_all();
721 while (all != NULL) {
722 PurpleAccount *account = all->data;
724 if (purple_account_get_enabled(account, purple_core_get_ui()))
725 list = g_list_append(list, account);
727 all = all->next;
730 return list;
733 PurpleAccount *
734 purple_accounts_find(const char *name, const char *protocol_id)
736 PurpleAccount *account = NULL;
737 GList *l;
738 char *who;
740 g_return_val_if_fail(name != NULL, NULL);
741 g_return_val_if_fail(protocol_id != NULL, NULL);
743 for (l = purple_accounts_get_all(); l != NULL; l = l->next) {
744 account = (PurpleAccount *)l->data;
746 if (!purple_strequal(purple_account_get_protocol_id(account), protocol_id))
747 continue;
749 who = g_strdup(purple_normalize(account, name));
750 if (purple_strequal(purple_normalize(account, purple_account_get_username(account)), who)) {
751 g_free(who);
752 return account;
754 g_free(who);
757 return NULL;
760 void
761 purple_accounts_restore_current_statuses()
763 GList *l;
764 PurpleAccount *account;
766 /* If we're not connected to the Internet right now, we bail on this */
767 if (!purple_network_is_available())
769 purple_debug_warning("accounts", "Network not connected; skipping reconnect\n");
770 return;
773 for (l = purple_accounts_get_all(); l != NULL; l = l->next)
775 account = (PurpleAccount *)l->data;
777 if (purple_account_get_enabled(account, purple_core_get_ui()) &&
778 (purple_presence_is_online(purple_account_get_presence(account))))
780 purple_account_connect(account);
785 static PurpleAccountUiOps *
786 purple_account_ui_ops_copy(PurpleAccountUiOps *ops)
788 PurpleAccountUiOps *ops_new;
790 g_return_val_if_fail(ops != NULL, NULL);
792 ops_new = g_new(PurpleAccountUiOps, 1);
793 *ops_new = *ops;
795 return ops_new;
798 GType
799 purple_account_ui_ops_get_type(void)
801 static GType type = 0;
803 if (type == 0) {
804 type = g_boxed_type_register_static("PurpleAccountUiOps",
805 (GBoxedCopyFunc)purple_account_ui_ops_copy,
806 (GBoxedFreeFunc)g_free);
809 return type;
812 void
813 purple_accounts_set_ui_ops(PurpleAccountUiOps *ops)
815 account_ui_ops = ops;
818 PurpleAccountUiOps *
819 purple_accounts_get_ui_ops(void)
821 return account_ui_ops;
824 void *
825 purple_accounts_get_handle(void)
827 static int handle;
829 return &handle;
832 static void
833 signed_on_cb(PurpleConnection *gc,
834 gpointer unused)
836 PurpleAccount *account = purple_connection_get_account(gc);
837 purple_account_clear_current_error(account);
839 purple_signal_emit(purple_accounts_get_handle(), "account-signed-on",
840 account);
843 static void
844 signed_off_cb(PurpleConnection *gc,
845 gpointer unused)
847 PurpleAccount *account = purple_connection_get_account(gc);
849 purple_signal_emit(purple_accounts_get_handle(), "account-signed-off",
850 account);
853 static void
854 connection_error_cb(PurpleConnection *gc,
855 PurpleConnectionError type,
856 const gchar *description,
857 gpointer unused)
859 PurpleAccount *account;
860 PurpleConnectionErrorInfo *err;
862 account = purple_connection_get_account(gc);
864 g_return_if_fail(account != NULL);
866 err = g_new0(PurpleConnectionErrorInfo, 1);
868 err->type = type;
869 err->description = g_strdup(description);
871 _purple_account_set_current_error(account, err);
873 purple_signal_emit(purple_accounts_get_handle(), "account-connection-error",
874 account, type, description);
877 static void
878 password_migration_cb(PurpleAccount *account)
880 /* account may be NULL (means: all) */
882 purple_accounts_schedule_save();
885 void
886 purple_accounts_init(void)
888 void *handle = purple_accounts_get_handle();
889 void *conn_handle = purple_connections_get_handle();
891 purple_signal_register(handle, "account-connecting",
892 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
893 PURPLE_TYPE_ACCOUNT);
895 purple_signal_register(handle, "account-disabled",
896 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
897 PURPLE_TYPE_ACCOUNT);
899 purple_signal_register(handle, "account-enabled",
900 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
901 PURPLE_TYPE_ACCOUNT);
903 purple_signal_register(handle, "account-setting-info",
904 purple_marshal_VOID__POINTER_POINTER, G_TYPE_NONE, 2,
905 PURPLE_TYPE_ACCOUNT, G_TYPE_STRING);
907 purple_signal_register(handle, "account-set-info",
908 purple_marshal_VOID__POINTER_POINTER, G_TYPE_NONE, 2,
909 PURPLE_TYPE_ACCOUNT, G_TYPE_STRING);
911 purple_signal_register(handle, "account-created",
912 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
913 PURPLE_TYPE_ACCOUNT);
915 purple_signal_register(handle, "account-destroying",
916 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
917 PURPLE_TYPE_ACCOUNT);
919 purple_signal_register(handle, "account-added",
920 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
921 PURPLE_TYPE_ACCOUNT);
923 purple_signal_register(handle, "account-removed",
924 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
925 PURPLE_TYPE_ACCOUNT);
927 purple_signal_register(handle, "account-status-changing",
928 purple_marshal_VOID__POINTER_POINTER_POINTER,
929 G_TYPE_NONE, 3, PURPLE_TYPE_ACCOUNT,
930 PURPLE_TYPE_STATUS, PURPLE_TYPE_STATUS);
932 purple_signal_register(handle, "account-status-changed",
933 purple_marshal_VOID__POINTER_POINTER_POINTER,
934 G_TYPE_NONE, 3, PURPLE_TYPE_ACCOUNT,
935 PURPLE_TYPE_STATUS, PURPLE_TYPE_STATUS);
937 purple_signal_register(handle, "account-actions-changed",
938 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
939 PURPLE_TYPE_ACCOUNT);
941 purple_signal_register(handle, "account-alias-changed",
942 purple_marshal_VOID__POINTER_POINTER, G_TYPE_NONE, 2,
943 PURPLE_TYPE_ACCOUNT, G_TYPE_STRING);
945 purple_signal_register(handle, "account-authorization-requested",
946 purple_marshal_INT__POINTER_POINTER_POINTER,
947 G_TYPE_INT, 4, PURPLE_TYPE_ACCOUNT, G_TYPE_STRING,
948 G_TYPE_STRING, G_TYPE_STRING);
950 purple_signal_register(handle, "account-authorization-denied",
951 purple_marshal_VOID__POINTER_POINTER, G_TYPE_NONE, 3,
952 PURPLE_TYPE_ACCOUNT, G_TYPE_STRING, G_TYPE_STRING);
954 purple_signal_register(handle, "account-authorization-granted",
955 purple_marshal_VOID__POINTER_POINTER, G_TYPE_NONE, 3,
956 PURPLE_TYPE_ACCOUNT, G_TYPE_STRING, G_TYPE_STRING);
958 purple_signal_register(handle, "account-error-changed",
959 purple_marshal_VOID__POINTER_POINTER_POINTER,
960 G_TYPE_NONE, 3, PURPLE_TYPE_ACCOUNT,
961 PURPLE_TYPE_CONNECTION_ERROR_INFO,
962 PURPLE_TYPE_CONNECTION_ERROR_INFO);
964 purple_signal_register(handle, "account-signed-on",
965 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
966 PURPLE_TYPE_ACCOUNT);
968 purple_signal_register(handle, "account-signed-off",
969 purple_marshal_VOID__POINTER, G_TYPE_NONE, 1,
970 PURPLE_TYPE_ACCOUNT);
972 purple_signal_register(handle, "account-connection-error",
973 purple_marshal_VOID__POINTER_INT_POINTER,
974 G_TYPE_NONE, 3, PURPLE_TYPE_ACCOUNT,
975 PURPLE_TYPE_CONNECTION_ERROR, G_TYPE_STRING);
977 purple_signal_connect(conn_handle, "signed-on", handle,
978 PURPLE_CALLBACK(signed_on_cb), NULL);
979 purple_signal_connect(conn_handle, "signed-off", handle,
980 PURPLE_CALLBACK(signed_off_cb), NULL);
981 purple_signal_connect(conn_handle, "connection-error", handle,
982 PURPLE_CALLBACK(connection_error_cb), NULL);
983 purple_signal_connect(purple_keyring_get_handle(), "password-migration", handle,
984 PURPLE_CALLBACK(password_migration_cb), NULL);
986 load_accounts();
990 void
991 purple_accounts_uninit(void)
993 gpointer handle = purple_accounts_get_handle();
994 if (save_timer != 0)
996 g_source_remove(save_timer);
997 save_timer = 0;
998 sync_accounts();
1001 for (; accounts; accounts = g_list_delete_link(accounts, accounts))
1002 g_object_unref(G_OBJECT(accounts->data));
1004 purple_signals_disconnect_by_handle(handle);
1005 purple_signals_unregister_by_instance(handle);