1 /* Copyright (c) 2006, Nokia Corporation
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * * Neither the name of the Nokia Corporation nor the names of its
14 * contributors may be used to endorse or promote products derived from
15 * this software without specific prior written permission.
17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
18 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
20 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
21 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
24 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
25 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
26 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 #include <modest-marshal.h>
32 #include <modest-account-mgr.h>
33 #include <modest-account-mgr-priv.h>
34 #include <modest-account-mgr-helpers.h>
36 /* 'private'/'protected' functions */
37 static void modest_account_mgr_class_init (ModestAccountMgrClass * klass);
38 static void modest_account_mgr_init (ModestAccountMgr * obj);
39 static void modest_account_mgr_finalize (GObject * obj);
43 ACCOUNT_CHANGED_SIGNAL,
44 ACCOUNT_REMOVED_SIGNAL,
51 static GObjectClass *parent_class = NULL;
52 static guint signals[LAST_SIGNAL] = {0};
54 /* We signal key changes in batches, every X seconds: */
56 on_timeout_notify_changes (gpointer data)
58 ModestAccountMgr *self = MODEST_ACCOUNT_MGR (data);
59 ModestAccountMgrPrivate *priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
61 /* TODO: Also store the account names, and notify one list for each account,
62 * if anything uses the account names. */
64 if (priv->changed_conf_keys) {
65 gchar *default_account =
66 modest_account_mgr_get_default_account (self);
68 g_signal_emit (G_OBJECT(self), signals[ACCOUNT_CHANGED_SIGNAL], 0,
69 default_account, priv->changed_conf_keys, FALSE);
71 g_free (default_account);
73 g_slist_foreach (priv->changed_conf_keys, (GFunc) g_free, NULL);
74 g_slist_free (priv->changed_conf_keys);
75 priv->changed_conf_keys = NULL;
78 return TRUE; /* Call this again later. */
82 on_key_change (ModestConf *conf, const gchar *key, ModestConfEvent event, gpointer user_data)
84 ModestAccountMgr *self = MODEST_ACCOUNT_MGR (user_data);
85 ModestAccountMgrPrivate *priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
86 gboolean is_account_key;
87 gboolean is_server_account;
88 gchar* account = NULL;
90 /* there is only one not-really-account key which will still emit
91 * a signal: a change in MODEST_CONF_DEFAULT_ACCOUNT */
92 if (key && strcmp (key, MODEST_CONF_DEFAULT_ACCOUNT) == 0) {
93 /* Get the default account instead. */
95 /* Store the key for later notification in our timeout callback.
96 * Notifying for every key change would cause unnecessary work: */
97 priv->changed_conf_keys = g_slist_append (priv->changed_conf_keys,
98 (gpointer) g_strdup (key));
101 is_account_key = FALSE;
102 is_server_account = FALSE;
103 account = _modest_account_mgr_account_from_key (key, &is_account_key,
106 /* if this is not an account-related key change, ignore */
110 /* account was removed. Do not emit an account removed signal
111 because it was already being done in the remove_account
112 method. Do not notify also the removal of the server
113 account keys for the same reason */
114 if ((is_account_key || is_server_account) &&
115 event == MODEST_CONF_EVENT_KEY_UNSET) {
120 /* is this account enabled? */
121 gboolean enabled = FALSE;
122 if (is_server_account)
125 enabled = modest_account_mgr_get_enabled (self, account);
127 /* Notify is server account was changed, default account was changed
128 * or when enabled/disabled changes:
131 g_str_has_suffix (key, MODEST_ACCOUNT_ENABLED) ||
132 strcmp (key, MODEST_CONF_DEFAULT_ACCOUNT) == 0) {
133 /* Store the key for later notification in our timeout callback.
134 * Notifying for every key change would cause unnecessary work: */
135 priv->changed_conf_keys = g_slist_append (NULL,
136 (gpointer) g_strdup (key));
143 modest_account_mgr_get_type (void)
145 static GType my_type = 0;
148 static const GTypeInfo my_info = {
149 sizeof (ModestAccountMgrClass),
150 NULL, /* base init */
151 NULL, /* base finalize */
152 (GClassInitFunc) modest_account_mgr_class_init,
153 NULL, /* class finalize */
154 NULL, /* class data */
155 sizeof (ModestAccountMgr),
157 (GInstanceInitFunc) modest_account_mgr_init,
161 my_type = g_type_register_static (G_TYPE_OBJECT,
169 modest_account_mgr_class_init (ModestAccountMgrClass * klass)
171 GObjectClass *gobject_class;
172 gobject_class = (GObjectClass *) klass;
174 parent_class = g_type_class_peek_parent (klass);
175 gobject_class->finalize = modest_account_mgr_finalize;
177 g_type_class_add_private (gobject_class,
178 sizeof (ModestAccountMgrPrivate));
180 /* signal definitions */
181 signals[ACCOUNT_REMOVED_SIGNAL] =
182 g_signal_new ("account_removed",
183 G_TYPE_FROM_CLASS (klass),
185 G_STRUCT_OFFSET(ModestAccountMgrClass,account_removed),
187 modest_marshal_VOID__STRING_BOOLEAN,
188 G_TYPE_NONE, 2, G_TYPE_STRING, G_TYPE_BOOLEAN);
189 signals[ACCOUNT_CHANGED_SIGNAL] =
190 g_signal_new ("account_changed",
191 G_TYPE_FROM_CLASS (klass),
193 G_STRUCT_OFFSET(ModestAccountMgrClass,account_changed),
195 modest_marshal_VOID__STRING_POINTER_BOOLEAN,
196 G_TYPE_NONE, 3, G_TYPE_STRING, G_TYPE_POINTER, G_TYPE_BOOLEAN);
197 signals[ACCOUNT_BUSY_SIGNAL] =
198 g_signal_new ("account_busy_changed",
199 G_TYPE_FROM_CLASS (klass),
201 G_STRUCT_OFFSET(ModestAccountMgrClass,account_busy_changed),
203 modest_marshal_VOID__STRING_BOOLEAN,
204 G_TYPE_NONE, 2, G_TYPE_STRING, G_TYPE_BOOLEAN);
209 modest_account_mgr_init (ModestAccountMgr * obj)
211 ModestAccountMgrPrivate *priv =
212 MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
214 priv->modest_conf = NULL;
215 priv->busy_accounts = NULL;
216 priv->timeout = g_timeout_add (1000 /* milliseconds */, on_timeout_notify_changes, obj);
220 modest_account_mgr_finalize (GObject * obj)
222 ModestAccountMgrPrivate *priv =
223 MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
225 if (priv->key_changed_handler_uid) {
226 g_signal_handler_disconnect (priv->modest_conf,
227 priv->key_changed_handler_uid);
228 priv->key_changed_handler_uid = 0;
231 if (priv->modest_conf) {
232 g_object_unref (G_OBJECT(priv->modest_conf));
233 priv->modest_conf = NULL;
237 g_source_remove (priv->timeout);
239 if (priv->changed_conf_keys) {
240 g_slist_foreach (priv->changed_conf_keys, (GFunc) g_free, NULL);
241 g_slist_free (priv->changed_conf_keys);
244 G_OBJECT_CLASS(parent_class)->finalize (obj);
249 modest_account_mgr_new (ModestConf *conf)
252 ModestAccountMgrPrivate *priv;
254 g_return_val_if_fail (conf, NULL);
256 obj = G_OBJECT (g_object_new (MODEST_TYPE_ACCOUNT_MGR, NULL));
257 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
259 g_object_ref (G_OBJECT(conf));
260 priv->modest_conf = conf;
262 priv->key_changed_handler_uid =
263 g_signal_connect (G_OBJECT (conf), "key_changed",
264 G_CALLBACK (on_key_change),
267 return MODEST_ACCOUNT_MGR (obj);
272 null_means_empty (const gchar * str)
274 return str ? str : "";
279 modest_account_mgr_add_account (ModestAccountMgr *self,
281 const gchar *store_account,
282 const gchar *transport_account,
285 ModestAccountMgrPrivate *priv;
288 gchar *default_account;
291 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
292 g_return_val_if_fail (name, FALSE);
293 g_return_val_if_fail (strchr(name, '/') == NULL, FALSE);
295 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
298 * we create the account by adding an account 'dir', with the name <name>,
299 * and in that the 'display_name' string key
301 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_DISPLAY_NAME, FALSE);
302 if (modest_account_mgr_account_exists (self, key, FALSE)) {
303 g_printerr ("modest: account already exists\n");
308 ok = modest_conf_set_string (priv->modest_conf, key, name, &err);
311 g_printerr ("modest: cannot set display name\n");
313 g_printerr ("modest: Error adding account conf: %s\n", err->message);
320 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_STORE_ACCOUNT, FALSE);
321 ok = modest_conf_set_string (priv->modest_conf, key, store_account, &err);
324 g_printerr ("modest: failed to set store account '%s'\n",
327 g_printerr ("modest: Error adding store account conf: %s\n", err->message);
334 if (transport_account) {
335 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_TRANSPORT_ACCOUNT,
337 ok = modest_conf_set_string (priv->modest_conf, key, transport_account, &err);
340 g_printerr ("modest: failed to set transport account '%s'\n",
343 g_printerr ("modest: Error adding transport account conf: %s\n", err->message);
350 /* Make sure that leave-messages-on-server is enabled by default,
351 * as per the UI spec, though it is only meaningful for accounts using POP.
352 * (possibly this gconf key should be under the server account): */
353 modest_account_mgr_set_bool (self, name,
354 MODEST_ACCOUNT_LEAVE_ON_SERVER, TRUE, FALSE /* not server account */);
357 modest_account_mgr_set_enabled (self, name, enabled);
359 /* if no default account has been defined yet, do so now */
360 default_account = modest_account_mgr_get_default_account (self);
361 if (!default_account)
362 modest_account_mgr_set_default_account (self, name);
363 g_free (default_account);
370 modest_account_mgr_add_server_account (ModestAccountMgr * self,
371 const gchar * name, const gchar *hostname,
373 const gchar * username, const gchar * password,
374 ModestTransportStoreProtocol proto,
375 ModestConnectionProtocol security,
376 ModestAuthProtocol auth)
378 ModestAccountMgrPrivate *priv;
383 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
384 g_return_val_if_fail (name, FALSE);
385 g_return_val_if_fail (strchr(name, '/') == NULL, FALSE);
387 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
390 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_HOSTNAME, TRUE);
391 if (modest_conf_key_exists (priv->modest_conf, key, &err)) {
392 g_printerr ("modest: server account '%s' already exists\n", name);
399 modest_conf_set_string (priv->modest_conf, key, null_means_empty(hostname), &err);
401 g_printerr ("modest: failed to set %s: %s\n", key, err->message);
410 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_USERNAME, TRUE);
411 ok = modest_conf_set_string (priv->modest_conf, key, null_means_empty (username), &err);
413 g_printerr ("modest: failed to set %s: %s\n", key, err->message);
423 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_PASSWORD, TRUE);
424 ok = modest_conf_set_string (priv->modest_conf, key, null_means_empty (password), &err);
426 g_printerr ("modest: failed to set %s: %s\n", key, err->message);
435 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_PROTO, TRUE);
436 ok = modest_conf_set_string (priv->modest_conf, key,
437 modest_protocol_info_get_transport_store_protocol_name(proto),
440 g_printerr ("modest: failed to set %s: %s\n", key, err->message);
450 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_PORT, TRUE);
451 ok = modest_conf_set_int (priv->modest_conf, key, portnumber, &err);
453 g_printerr ("modest: failed to set %s: %s\n", key, err->message);
463 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_AUTH_MECH, TRUE);
464 ok = modest_conf_set_string (priv->modest_conf, key,
465 modest_protocol_info_get_auth_protocol_name (auth),
468 g_printerr ("modest: failed to set %s: %s\n", key, err->message);
476 /* Add the security settings: */
477 modest_server_account_set_security (self, name, security);
481 g_printerr ("modest: failed to add server account\n");
488 /** modest_account_mgr_add_server_account_uri:
489 * Only used for mbox and maildir accounts.
492 modest_account_mgr_add_server_account_uri (ModestAccountMgr * self,
493 const gchar *name, ModestTransportStoreProtocol proto,
496 ModestAccountMgrPrivate *priv;
500 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
501 g_return_val_if_fail (name, FALSE);
502 g_return_val_if_fail (strchr(name, '/') == NULL, FALSE);
503 g_return_val_if_fail (uri, FALSE);
505 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
509 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_PROTO, TRUE);
510 ok = modest_conf_set_string (priv->modest_conf, key,
511 modest_protocol_info_get_transport_store_protocol_name(proto),
516 g_printerr ("modest: failed to set proto\n");
521 key = _modest_account_mgr_get_account_keyname (name, MODEST_ACCOUNT_URI, TRUE);
522 ok = modest_conf_set_string (priv->modest_conf, key, uri, NULL);
526 g_printerr ("modest: failed to set uri\n");
533 modest_account_mgr_remove_account (ModestAccountMgr * self,
534 const gchar* name, gboolean server_account)
536 ModestAccountMgrPrivate *priv;
541 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
542 g_return_val_if_fail (name, FALSE);
544 if (!modest_account_mgr_account_exists (self, name, server_account)) {
545 g_printerr ("modest: %s: account '%s' does not exist\n", __FUNCTION__, name);
549 if (!server_account) {
550 gchar *server_account_name;
552 /* in case we're deleting an account, also delete the dependent store and transport account */
553 server_account_name = modest_account_mgr_get_string (self, name, MODEST_ACCOUNT_STORE_ACCOUNT,
555 if (server_account_name) {
556 if (!modest_account_mgr_remove_account (self, server_account_name, TRUE))
557 g_printerr ("modest: failed to remove store account '%s' (%s)\n",
558 server_account_name, name);
559 g_free (server_account_name);
561 g_printerr ("modest: could not find the store account for %s\n", name);
563 server_account_name = modest_account_mgr_get_string (self, name, MODEST_ACCOUNT_TRANSPORT_ACCOUNT,
565 if (server_account_name) {
566 if (!modest_account_mgr_remove_account (self, server_account_name, TRUE))
567 g_printerr ("modest: failed to remove transport account '%s' (%s)\n",
568 server_account_name, name);
569 g_free (server_account_name);
571 g_printerr ("modest: could not find the transport account for %s\n", name);
574 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
575 key = _modest_account_mgr_get_account_keyname (name, NULL, server_account);
577 retval = modest_conf_remove_key (priv->modest_conf, key, &err);
581 g_printerr ("modest: error removing key: %s\n", err->message);
585 /* If this was the default, then remove that setting: */
586 if (!server_account) {
587 gchar *default_account_name = modest_account_mgr_get_default_account (self);
588 if (default_account_name && (strcmp (default_account_name, name) == 0))
589 modest_account_mgr_unset_default_account (self);
590 g_free (default_account_name);
592 /* pick another one as the new default account */
593 modest_account_mgr_set_first_account_as_default (self);
595 /* Notify the observers. We do this *after* deleting
596 the keys, because otherwise a call to account_names
597 will retrieve also the deleted account */
598 g_signal_emit (G_OBJECT(self), signals[ACCOUNT_REMOVED_SIGNAL], 0,
599 name, server_account);
606 /* strip the first /n/ character from each element
607 * caller must make sure all elements are strings with
608 * length >= n, and also that data can be freed.
612 strip_prefix_from_elements (GSList * lst, guint n)
615 memmove (lst->data, lst->data + n,
616 strlen(lst->data) - n + 1);
623 modest_account_mgr_account_names (ModestAccountMgr * self, gboolean only_enabled)
626 ModestAccountMgrPrivate *priv;
629 const size_t prefix_len = strlen (MODEST_ACCOUNT_NAMESPACE "/");
631 g_return_val_if_fail (self, NULL);
633 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
634 accounts = modest_conf_list_subkeys (priv->modest_conf,
635 MODEST_ACCOUNT_NAMESPACE, &err);
638 g_printerr ("modest: failed to get subkeys (%s): %s\n",
639 MODEST_ACCOUNT_NAMESPACE, err->message);
641 return NULL; /* assume accounts did not get value when err is set...*/
644 strip_prefix_from_elements (accounts, prefix_len);
646 GSList *result = NULL;
648 /* Unescape the keys to get the account names: */
649 GSList *iter = accounts;
654 const gchar* account_name_key = (const gchar*)iter->data;
655 gchar* unescaped_name = account_name_key ?
656 modest_conf_key_unescape (account_name_key)
661 if (unescaped_name &&
662 !modest_account_mgr_get_enabled (self, unescaped_name)) {
668 result = g_slist_append (result, unescaped_name);
670 g_free (unescaped_name);
675 iter = g_slist_next (iter);
679 /* we already freed the strings in the loop */
680 g_slist_free (accounts);
688 modest_account_mgr_free_account_names (GSList *account_names)
690 g_slist_foreach (account_names, (GFunc)g_free, NULL);
691 g_slist_free (account_names);
697 modest_account_mgr_get_string (ModestAccountMgr *self, const gchar *name,
698 const gchar *key, gboolean server_account) {
700 ModestAccountMgrPrivate *priv;
706 g_return_val_if_fail (self, NULL);
707 g_return_val_if_fail (name, NULL);
708 g_return_val_if_fail (key, NULL);
710 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
712 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
713 retval = modest_conf_get_string (priv->modest_conf, keyname, &err);
715 g_printerr ("modest: error getting string '%s': %s\n", keyname, err->message);
726 modest_account_mgr_get_password (ModestAccountMgr *self, const gchar *name,
727 const gchar *key, gboolean server_account)
729 return modest_account_mgr_get_string (self, name, key, server_account);
736 modest_account_mgr_get_int (ModestAccountMgr *self, const gchar *name, const gchar *key,
737 gboolean server_account)
739 ModestAccountMgrPrivate *priv;
745 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), -1);
746 g_return_val_if_fail (name, -1);
747 g_return_val_if_fail (key, -1);
749 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
751 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
752 retval = modest_conf_get_int (priv->modest_conf, keyname, &err);
754 g_printerr ("modest: error getting int '%s': %s\n", keyname, err->message);
766 modest_account_mgr_get_bool (ModestAccountMgr * self, const gchar *account,
767 const gchar * key, gboolean server_account)
769 ModestAccountMgrPrivate *priv;
775 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
776 g_return_val_if_fail (account, FALSE);
777 g_return_val_if_fail (key, FALSE);
779 keyname = _modest_account_mgr_get_account_keyname (account, key, server_account);
781 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
782 retval = modest_conf_get_bool (priv->modest_conf, keyname, &err);
784 g_printerr ("modest: error getting bool '%s': %s\n", keyname, err->message);
796 modest_account_mgr_get_list (ModestAccountMgr *self, const gchar *name,
797 const gchar *key, ModestConfValueType list_type,
798 gboolean server_account)
800 ModestAccountMgrPrivate *priv;
806 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), NULL);
807 g_return_val_if_fail (name, NULL);
808 g_return_val_if_fail (key, NULL);
810 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
812 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
813 retval = modest_conf_get_list (priv->modest_conf, keyname, list_type, &err);
815 g_printerr ("modest: error getting list '%s': %s\n", keyname,
827 modest_account_mgr_set_string (ModestAccountMgr * self, const gchar * name,
828 const gchar * key, const gchar * val, gboolean server_account)
830 ModestAccountMgrPrivate *priv;
836 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
837 g_return_val_if_fail (name, FALSE);
838 g_return_val_if_fail (key, FALSE);
840 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
842 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
844 retval = modest_conf_set_string (priv->modest_conf, keyname, val, &err);
846 g_printerr ("modest: error setting string '%s': %s\n", keyname, err->message);
856 modest_account_mgr_set_password (ModestAccountMgr * self, const gchar * name,
857 const gchar * key, const gchar * val, gboolean server_account)
859 return modest_account_mgr_set_password (self, name, key, val, server_account);
865 modest_account_mgr_set_int (ModestAccountMgr * self, const gchar * name,
866 const gchar * key, int val, gboolean server_account)
868 ModestAccountMgrPrivate *priv;
874 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
875 g_return_val_if_fail (name, FALSE);
876 g_return_val_if_fail (key, FALSE);
878 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
880 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
882 retval = modest_conf_set_int (priv->modest_conf, keyname, val, &err);
884 g_printerr ("modest: error setting int '%s': %s\n", keyname, err->message);
895 modest_account_mgr_set_bool (ModestAccountMgr * self, const gchar * name,
896 const gchar * key, gboolean val, gboolean server_account)
898 ModestAccountMgrPrivate *priv;
904 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
905 g_return_val_if_fail (name, FALSE);
906 g_return_val_if_fail (key, FALSE);
908 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
910 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
912 retval = modest_conf_set_bool (priv->modest_conf, keyname, val, &err);
914 g_printerr ("modest: error setting bool '%s': %s\n", keyname, err->message);
924 modest_account_mgr_set_list (ModestAccountMgr *self,
928 ModestConfValueType list_type,
929 gboolean server_account)
931 ModestAccountMgrPrivate *priv;
936 g_return_val_if_fail (self, FALSE);
937 g_return_val_if_fail (name, FALSE);
938 g_return_val_if_fail (key, FALSE);
939 g_return_val_if_fail (val, FALSE);
941 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
943 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
944 retval = modest_conf_set_list (priv->modest_conf, keyname, val, list_type, &err);
946 g_printerr ("modest: error setting list '%s': %s\n", keyname, err->message);
956 modest_account_mgr_account_exists (ModestAccountMgr * self, const gchar * name,
957 gboolean server_account)
959 ModestAccountMgrPrivate *priv;
965 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
966 g_return_val_if_fail (name, FALSE);
968 keyname = _modest_account_mgr_get_account_keyname (name, NULL, server_account);
969 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
970 retval = modest_conf_key_exists (priv->modest_conf, keyname, &err);
972 g_printerr ("modest: error determining existance of '%s': %s\n", keyname,
982 modest_account_mgr_account_with_display_name_exists (ModestAccountMgr *self, const gchar *display_name)
984 GSList *account_names = NULL;
985 GSList *cursor = NULL;
987 cursor = account_names = modest_account_mgr_account_names (self,
988 TRUE /* enabled accounts, because disabled accounts are not user visible. */);
990 gboolean found = FALSE;
992 /* Look at each non-server account to check their display names; */
994 const gchar * account_name = (gchar*)cursor->data;
996 ModestAccountData *account_data = modest_account_mgr_get_account_data (self, account_name);
998 g_printerr ("modest: failed to get account data for %s\n", account_name);
1002 if(account_data->display_name && (strcmp (account_data->display_name, display_name) == 0)) {
1007 modest_account_mgr_free_account_data (self, account_data);
1008 cursor = cursor->next;
1010 modest_account_mgr_free_account_names (account_names);
1011 account_names = NULL;
1020 modest_account_mgr_unset (ModestAccountMgr *self, const gchar *name,
1021 const gchar *key, gboolean server_account)
1023 ModestAccountMgrPrivate *priv;
1029 g_return_val_if_fail (MODEST_IS_ACCOUNT_MGR(self), FALSE);
1030 g_return_val_if_fail (name, FALSE);
1031 g_return_val_if_fail (key, FALSE);
1033 keyname = _modest_account_mgr_get_account_keyname (name, key, server_account);
1035 priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
1036 retval = modest_conf_remove_key (priv->modest_conf, keyname, &err);
1038 g_printerr ("modest: error unsetting'%s': %s\n", keyname,
1048 _modest_account_mgr_account_from_key (const gchar *key, gboolean *is_account_key, gboolean *is_server_account)
1050 /* Initialize input parameters: */
1052 *is_account_key = FALSE;
1054 if (is_server_account)
1055 *is_server_account = FALSE;
1057 const gchar* account_ns = MODEST_ACCOUNT_NAMESPACE "/";
1058 const gchar* server_account_ns = MODEST_SERVER_ACCOUNT_NAMESPACE "/";
1060 gchar *account = NULL;
1062 /* determine whether it's an account or a server account,
1063 * based on the prefix */
1064 if (g_str_has_prefix (key, account_ns)) {
1066 if (is_server_account)
1067 *is_server_account = FALSE;
1069 account = g_strdup (key + strlen (account_ns));
1071 } else if (g_str_has_prefix (key, server_account_ns)) {
1073 if (is_server_account)
1074 *is_server_account = TRUE;
1076 account = g_strdup (key + strlen (server_account_ns));
1080 /* if there are any slashes left in the key, it's not
1081 * the toplevel entry for an account
1083 cursor = strstr(account, "/");
1085 if (is_account_key && cursor)
1086 *is_account_key = TRUE;
1088 /* put a NULL where the first slash was */
1093 /* The key is an escaped string, so unescape it to get the actual account name: */
1094 gchar *unescaped_name = modest_conf_key_unescape (account);
1096 return unescaped_name;
1103 /* must be freed by caller */
1105 _modest_account_mgr_get_account_keyname (const gchar *account_name, const gchar * name, gboolean server_account)
1107 gchar *retval = NULL;
1109 gchar *namespace = server_account ? MODEST_SERVER_ACCOUNT_NAMESPACE : MODEST_ACCOUNT_NAMESPACE;
1112 return g_strdup (namespace);
1114 /* Always escape the conf keys, so that it is acceptable to gconf: */
1115 gchar *escaped_account_name = account_name ? modest_conf_key_escape (account_name) : NULL;
1116 gchar *escaped_name = name ? modest_conf_key_escape (name) : NULL;
1118 if (escaped_account_name && escaped_name)
1119 retval = g_strconcat (namespace, "/", escaped_account_name, "/", escaped_name, NULL);
1120 else if (escaped_account_name)
1121 retval = g_strconcat (namespace, "/", escaped_account_name, NULL);
1124 if (!modest_conf_key_is_valid (retval)) {
1125 g_warning ("%s: Generated conf key was invalid: %s", __FUNCTION__, retval);
1130 g_free (escaped_name);
1131 g_free (escaped_account_name);
1137 modest_account_mgr_has_accounts (ModestAccountMgr* self, gboolean enabled)
1139 /* Check that at least one account exists: */
1140 GSList *account_names = modest_account_mgr_account_names (self,
1142 gboolean accounts_exist = account_names != NULL;
1144 modest_account_mgr_free_account_names (account_names);
1145 account_names = NULL;
1147 return accounts_exist;
1151 compare_account_name(gconstpointer a, gconstpointer b)
1153 const gchar* account_name = (const gchar*) a;
1154 const gchar* account_name2 = (const gchar*) b;
1155 return strcmp(account_name, account_name2);
1159 modest_account_mgr_set_account_busy(ModestAccountMgr* self, const gchar* account_name,
1162 ModestAccountMgrPrivate* priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
1165 GSList *account_names = modest_account_mgr_account_names (self,
1168 g_slist_find_custom(account_names, account_name, (GCompareFunc) compare_account_name);
1169 if (account && !modest_account_mgr_account_is_busy(self, account_name))
1171 priv->busy_accounts = g_slist_append(priv->busy_accounts, g_strdup(account_name));
1172 g_signal_emit_by_name(G_OBJECT(self), "account-busy-changed", account_name, TRUE);
1174 modest_account_mgr_free_account_names (account_names);
1175 account_names = NULL;
1178 g_slist_find_custom(priv->busy_accounts, account_name, (GCompareFunc) compare_account_name);
1181 g_free(account->data);
1182 priv->busy_accounts = g_slist_delete_link(priv->busy_accounts, account);
1183 g_signal_emit_by_name(G_OBJECT(self), "account-busy-changed", account_name, FALSE);
1189 modest_account_mgr_account_is_busy(ModestAccountMgr* self, const gchar* account_name)
1191 ModestAccountMgrPrivate* priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
1192 return (g_slist_find_custom(priv->busy_accounts, account_name, (GCompareFunc) compare_account_name)