* cosmetics
[modest] / src / modest-account-mgr.c
1 /* Copyright (c) 2006, Nokia Corporation
2  * All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
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.
16  *
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.
28  */
29
30 #include <string.h>
31 #include "modest-marshal.h"
32 #include "modest-account-keys.h"
33 #include "modest-account-mgr.h"
34
35 /* 'private'/'protected' functions */
36 static void modest_account_mgr_class_init (ModestAccountMgrClass * klass);
37 static void modest_account_mgr_init (ModestAccountMgr * obj);
38 static void modest_account_mgr_finalize (GObject * obj);
39
40 static gchar *get_account_keyname (const gchar * accname, const gchar * name, gboolean server_account);
41
42 /* list my signals */
43 enum {
44         ACCOUNT_CHANGED_SIGNAL,
45         ACCOUNT_REMOVED_SIGNAL,
46         LAST_SIGNAL
47 };
48
49 typedef struct _ModestAccountMgrPrivate ModestAccountMgrPrivate;
50 struct _ModestAccountMgrPrivate {
51         ModestConf *modest_conf;
52         GSList *current_accounts;
53 };
54
55 #define MODEST_ACCOUNT_MGR_GET_PRIVATE(o)      (G_TYPE_INSTANCE_GET_PRIVATE((o), \
56                                                 MODEST_TYPE_ACCOUNT_MGR, \
57                                                 ModestAccountMgrPrivate))
58 /* globals */
59 static GObjectClass *parent_class = NULL;
60
61 static guint signals[LAST_SIGNAL] = {0};
62
63
64 static gchar*
65 account_from_key (const gchar *key, gboolean *is_account_key, gboolean *is_server_account)
66 {
67         const gchar* account_ns        = MODEST_ACCOUNT_NAMESPACE "/";
68         const gchar* server_account_ns = MODEST_SERVER_ACCOUNT_NAMESPACE "/";
69         gchar *cursor;
70         gchar *account = NULL;
71
72         /* determine if it's an account or a server account,
73          * based on the prefix */
74         if (g_str_has_prefix (key, account_ns)) {
75
76                 if (is_server_account)
77                         *is_server_account = FALSE;
78                 
79                 account = g_strdup (key + strlen (account_ns));
80
81         } else if (g_str_has_prefix (key, server_account_ns)) {
82
83                 if (is_server_account)
84                         *is_server_account = TRUE;
85                 
86                 account = g_strdup (key + strlen (server_account_ns));  
87         } else
88                 return NULL;
89
90         /* if there are any slashes left in the key, it's not
91          * the toplevel entry for an account
92          */
93         cursor = strstr(account, "/");
94         
95         if (is_account_key && cursor)
96                 *is_account_key = TRUE;
97
98         /* put a NULL where the first slash was */
99         if (cursor)
100                 *cursor = '\0';
101                 
102         return account;
103 }
104
105
106
107
108 static void
109 on_key_change (ModestConf *conf, const gchar *key, ModestConfEvent event, gpointer user_data)
110 {
111         ModestAccountMgr *self;
112         ModestAccountMgrPrivate *priv;
113
114         gchar *account;
115         gboolean is_account_key, is_server_account;
116         gboolean enabled;
117
118         self = MODEST_ACCOUNT_MGR (user_data);
119         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
120         
121         account = account_from_key (key, &is_account_key, &is_server_account);
122
123         /* account was removed -- emit this, even if the account was disabled */
124         if (is_account_key && event == MODEST_CONF_EVENT_KEY_UNSET) {
125                 g_signal_emit (G_OBJECT(self), signals[ACCOUNT_REMOVED_SIGNAL], 0,
126                                account, is_server_account);
127                 g_free (account);
128                 return;
129         }
130
131         /* is this account enabled? */
132         enabled = modest_account_mgr_account_get_enabled (self, account,
133                                                           is_server_account);
134
135         /* account was changed.
136          * and always notify when enabled/disabled changes
137          */
138         if (enabled || g_str_has_suffix (key, MODEST_ACCOUNT_ENABLED)) 
139                 g_signal_emit (G_OBJECT(self), signals[ACCOUNT_CHANGED_SIGNAL], 0,
140                                account, key, is_server_account);
141
142         g_free (account);
143 }
144
145
146 GType
147 modest_account_mgr_get_type (void)
148 {
149         static GType my_type = 0;
150
151         if (!my_type) {
152                 static const GTypeInfo my_info = {
153                         sizeof (ModestAccountMgrClass),
154                         NULL,   /* base init */
155                         NULL,   /* base finalize */
156                         (GClassInitFunc) modest_account_mgr_class_init,
157                         NULL,   /* class finalize */
158                         NULL,   /* class data */
159                         sizeof (ModestAccountMgr),
160                         1,      /* n_preallocs */
161                         (GInstanceInitFunc) modest_account_mgr_init,
162                 };
163
164                 my_type = g_type_register_static (G_TYPE_OBJECT,
165                                                   "ModestAccountMgr",
166                                                   &my_info, 0);
167         }
168         return my_type;
169 }
170
171 static void
172 modest_account_mgr_class_init (ModestAccountMgrClass * klass)
173 {
174         GObjectClass *gobject_class;
175         GType paramtypes[3] = {G_TYPE_POINTER, G_TYPE_POINTER, G_TYPE_POINTER};
176
177         gobject_class = (GObjectClass *) klass;
178
179         parent_class = g_type_class_peek_parent (klass);
180         gobject_class->finalize = modest_account_mgr_finalize;
181
182         g_type_class_add_private (gobject_class,
183                                   sizeof (ModestAccountMgrPrivate));
184
185         /* signal definitions */
186         signals[ACCOUNT_REMOVED_SIGNAL] =
187                 g_signal_new ("account_removed",
188                               G_TYPE_FROM_CLASS (klass),
189                               G_SIGNAL_RUN_FIRST,
190                               G_STRUCT_OFFSET(ModestAccountMgrClass,account_removed),
191                               NULL, NULL,
192                               modest_marshal_VOID__STRING_BOOLEAN,
193                               G_TYPE_NONE, 2, G_TYPE_STRING, G_TYPE_BOOLEAN);
194         signals[ACCOUNT_CHANGED_SIGNAL] =
195                 g_signal_new ("account_changed",
196                                G_TYPE_FROM_CLASS (klass),
197                               G_SIGNAL_RUN_FIRST,
198                               G_STRUCT_OFFSET(ModestAccountMgrClass,account_changed),
199                               NULL, NULL,
200                               modest_marshal_VOID__STRING_STRING_BOOLEAN,
201                               G_TYPE_NONE, 3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN);
202 }
203
204
205 static void
206 modest_account_mgr_init (ModestAccountMgr * obj)
207 {
208         ModestAccountMgrPrivate *priv =
209                 MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
210
211         priv->modest_conf = NULL;
212 }
213
214 static void
215 modest_account_mgr_finalize (GObject * obj)
216 {
217         ModestAccountMgrPrivate *priv =
218                 MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
219
220         if (priv->modest_conf) {
221                 g_object_unref (G_OBJECT(priv->modest_conf));
222                 priv->modest_conf = NULL;
223         }
224 }
225
226
227 ModestAccountMgr *
228 modest_account_mgr_new (ModestConf * conf)
229 {
230         GObject *obj;
231         ModestAccountMgrPrivate *priv;
232
233         g_return_val_if_fail (conf, NULL);
234
235         obj = G_OBJECT (g_object_new (MODEST_TYPE_ACCOUNT_MGR, NULL));
236         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
237
238         g_object_ref (G_OBJECT(conf));
239         priv->modest_conf = conf;
240
241         g_signal_connect (G_OBJECT (conf), "key_changed",
242                           G_CALLBACK (on_key_change),
243                           obj);
244         
245         return MODEST_ACCOUNT_MGR (obj);
246 }
247
248
249 static const gchar *
250 null_means_empty (const gchar * str)
251 {
252         return str ? str : "";
253 }
254
255
256 gboolean
257 modest_account_mgr_account_set_enabled (ModestAccountMgr *self, const gchar* name,
258                                         gboolean is_server_account, gboolean enabled)
259 {
260         return modest_account_mgr_set_bool (self, name,
261                                             MODEST_ACCOUNT_ENABLED, enabled,
262                                             is_server_account, NULL);
263 }
264
265
266 gboolean
267 modest_account_mgr_account_get_enabled (ModestAccountMgr *self, const gchar* name,
268                                         gboolean is_server_account)
269 {
270         return modest_account_mgr_get_bool (self, name,
271                                             MODEST_ACCOUNT_ENABLED, is_server_account,
272                                             NULL);
273 }
274
275
276 gboolean
277 modest_account_mgr_add_account (ModestAccountMgr *self,
278                                 const gchar *name,
279                                 const gchar *store_account,
280                                 const gchar *transport_account,
281                                 GError **err)
282 {
283         ModestAccountMgrPrivate *priv;
284         gchar *key;
285         gboolean ok;
286
287         g_return_val_if_fail (self, FALSE);
288         g_return_val_if_fail (name, FALSE);
289
290         if (modest_account_mgr_account_exists (self, name, FALSE, err)) {
291                 g_printerr ("modest: account already exists\n");
292                 return FALSE;
293         }
294         
295         /*
296          * we create the account by adding an account 'dir', with the name <name>,
297          * and in that the 'display_name' string key
298          */
299         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
300         
301         key = get_account_keyname (name, MODEST_ACCOUNT_DISPLAY_NAME, FALSE);
302         ok = modest_conf_set_string (priv->modest_conf, key, name, err);
303         g_free (key);
304
305         if (!ok) {
306                 g_printerr ("modest: cannot set display name\n");
307                 return FALSE;
308         }
309         
310         if (store_account) {
311                 key = get_account_keyname (name, MODEST_ACCOUNT_STORE_ACCOUNT, FALSE);
312                 ok = modest_conf_set_string (priv->modest_conf, key, store_account, err);
313                 g_free (key);
314                 if (!ok) {
315                         g_printerr ("modest: failed to set store account '%s'\n",
316                                 store_account);
317                         return FALSE;
318                 }
319         }
320
321         if (transport_account) {
322                 key = get_account_keyname (name, MODEST_ACCOUNT_TRANSPORT_ACCOUNT, FALSE);
323                 ok = modest_conf_set_string (priv->modest_conf, key, transport_account, err);
324                 g_free (key);
325                 if (!ok) {
326                         g_printerr ("modest: failed to set transport account '%s'\n",
327                                 transport_account);
328                         return FALSE;
329                 }
330         }
331
332         modest_account_mgr_account_set_enabled (self, name, FALSE, TRUE);
333         
334         return TRUE;
335 }
336
337
338
339
340 gboolean
341 modest_account_mgr_add_server_account (ModestAccountMgr * self,
342                                        const gchar * name,
343                                        const gchar * hostname,
344                                        const gchar * username,
345                                        const gchar * password,
346                                        const gchar * proto)
347 {
348         ModestAccountMgrPrivate *priv;
349         gchar *key;
350
351         g_return_val_if_fail (self, FALSE);
352         g_return_val_if_fail (name, FALSE);
353
354         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
355
356         key = get_account_keyname (name, NULL, TRUE);
357         if (modest_conf_key_exists (priv->modest_conf, key, NULL)) {
358                 g_printerr ("modest: server account '%s' already exists", name);
359                 g_free (key);
360                 return FALSE;
361         }
362         g_free (key);
363         
364         /* hostname */
365         key = get_account_keyname (name, MODEST_ACCOUNT_HOSTNAME, TRUE);
366         modest_conf_set_string (priv->modest_conf, key, null_means_empty(hostname), NULL);
367         g_free (key);
368
369         /* username */
370         key = get_account_keyname (name, MODEST_ACCOUNT_USERNAME, TRUE);
371         modest_conf_set_string (priv->modest_conf, key, null_means_empty (username), NULL);
372         g_free (key);
373
374         /* password */
375         key = get_account_keyname (name, MODEST_ACCOUNT_PASSWORD, TRUE);
376         modest_conf_set_string (priv->modest_conf, key, null_means_empty (password), NULL);
377         g_free (key);
378
379         /* proto */
380         key = get_account_keyname (name, MODEST_ACCOUNT_PROTO, TRUE);
381         modest_conf_set_string (priv->modest_conf, key, null_means_empty (proto), NULL);
382         g_free (key);
383
384         /* enable it */
385         modest_account_mgr_account_set_enabled (self, name, TRUE, TRUE);
386         
387         return TRUE;
388 }
389
390
391
392 gboolean
393 modest_account_mgr_remove_account (ModestAccountMgr * self,
394                                    const gchar * name,
395                                    gboolean server_account,
396                                    GError ** err)
397 {
398         ModestAccountMgrPrivate *priv;
399         gchar *key;
400         gboolean retval;
401
402         g_return_val_if_fail (self, FALSE);
403         g_return_val_if_fail (name, FALSE);
404
405         if (!modest_account_mgr_account_exists (self, name, server_account, err)) {
406                 g_printerr ("modest: account '%s' does not exist\n", name);
407                 return FALSE;
408         }
409
410         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
411         key = get_account_keyname (name, NULL, server_account);
412
413         retval = modest_conf_remove_key (priv->modest_conf, key, NULL);
414
415         g_free (key);
416         return retval;
417 }
418
419
420
421 /* strip the first /n/ character from each element */
422 /* caller must make sure all elements are strings with
423  * length >= n, and also that data can be freed.
424  */
425 static GSList*
426 strip_prefix_from_elements (GSList * lst, guint n)
427 {
428         GSList *cursor = lst;
429
430         while (cursor) {
431                 gchar *str = (gchar *) cursor->data;
432                 cursor->data = g_strdup (str + n);
433                 g_free (str);
434                 cursor = cursor->next;
435         }
436         return lst;
437 }
438
439
440 GSList *
441 modest_account_mgr_search_server_accounts (ModestAccountMgr * self,
442                                            const gchar * account_name,
443                                            ModestProtoType type,
444                                            const gchar *proto,
445                                            gboolean only_enabled)
446 {
447         GSList *accounts;
448         GSList *cursor;
449         ModestAccountMgrPrivate *priv;
450         gchar *key;
451         GError *err = NULL;
452         
453         g_return_val_if_fail (self, NULL);
454         
455         key      = get_account_keyname (account_name, NULL, TRUE);
456         priv     = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
457         
458         /* get the list of all server accounts */
459         accounts = modest_conf_list_subkeys (priv->modest_conf, key, &err);
460         if (err) {
461                 g_error_free (err);
462                 g_printerr ("modest: failed to get subkeys for '%s'\n", key);
463                 return NULL;
464         }
465         
466         /* no restrictions, return everything */
467         if (type == MODEST_PROTO_TYPE_ANY && !proto)
468                 return strip_prefix_from_elements (accounts, strlen(key)+1);
469         /* +1 because we must remove the ending '/' as well */
470         
471         /* otherwise, filter out the none-matching ones */
472         cursor = accounts;
473         while (cursor) {
474                 gchar *account;
475                 gchar *acc_proto;
476                 gboolean enabled;
477                 
478                 account = account_from_key ((gchar*)cursor->data, NULL, NULL);
479                 
480                 enabled   = modest_account_mgr_account_get_enabled (self, account, TRUE);
481                 acc_proto = modest_account_mgr_get_string (self, account, MODEST_ACCOUNT_PROTO,
482                                                            TRUE, NULL);
483                 if ((!acc_proto) ||                                /* proto not defined? */
484                     (type != MODEST_PROTO_TYPE_ANY &&              /* proto type ...     */
485                      modest_proto_type (acc_proto) != type) ||     /* ... matches?       */
486                     (proto && strcmp (proto, acc_proto) != 0) ||   /* proto matches?     */
487                     (!enabled && only_enabled)) {                  /* account enabled?   */
488                         /* match! remove from the list */
489                         GSList *nxt = cursor->next;
490                         accounts = g_slist_delete_link (accounts, cursor);
491                         cursor = nxt;
492                 } else
493                         cursor = cursor->next;
494
495                 g_free (account);
496                 g_free (acc_proto);
497         }
498
499         return strip_prefix_from_elements (accounts, strlen(key)+1);
500         /* +1 because we must remove the ending '/' as well */
501 }
502
503
504 GSList *
505 modest_account_mgr_account_names (ModestAccountMgr * self, GError ** err)
506 {
507         GSList *accounts;
508         ModestAccountMgrPrivate *priv;
509         const size_t prefix_len = strlen (MODEST_ACCOUNT_NAMESPACE "/");
510
511
512         g_return_val_if_fail (self, NULL);
513
514         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
515
516         accounts = modest_conf_list_subkeys (priv->modest_conf,
517                                              MODEST_ACCOUNT_NAMESPACE, err);
518         return strip_prefix_from_elements (accounts, prefix_len);
519 }
520
521
522 gchar *
523 modest_account_mgr_get_string (ModestAccountMgr *self, const gchar *name,
524                                const gchar *key, gboolean server_account, GError **err) {
525
526         ModestAccountMgrPrivate *priv;
527
528         gchar *keyname;
529         gchar *retval;
530
531         g_return_val_if_fail (self, NULL);
532         g_return_val_if_fail (name, NULL);
533         g_return_val_if_fail (key, NULL);
534
535         keyname = get_account_keyname (name, key, server_account);
536         
537         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
538         retval = modest_conf_get_string (priv->modest_conf, keyname, err);
539         g_free (keyname);
540
541         return retval;
542 }
543
544
545 gint
546 modest_account_mgr_get_int (ModestAccountMgr *self, const gchar *name,
547                             const gchar *key, gboolean server_account, GError **err)
548 {
549         ModestAccountMgrPrivate *priv;
550
551         gchar *keyname;
552         gint retval;
553
554         g_return_val_if_fail (self, -1);
555         g_return_val_if_fail (name, -1);
556         g_return_val_if_fail (key, -1);
557
558         keyname = get_account_keyname (name, key, server_account);
559
560         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
561         retval = modest_conf_get_int (priv->modest_conf, keyname, err);
562         g_free (keyname);
563
564         return retval;
565 }
566
567
568
569 gboolean
570 modest_account_mgr_get_bool (ModestAccountMgr * self, const gchar *account,
571                              const gchar * key, gboolean server_account, GError ** err)
572 {
573         ModestAccountMgrPrivate *priv;
574
575         gchar *keyname;
576         gboolean retval;
577
578         g_return_val_if_fail (self, FALSE);
579         g_return_val_if_fail (account, FALSE);
580         g_return_val_if_fail (key, FALSE);
581
582         keyname = get_account_keyname (account, key, server_account);
583         
584         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
585         retval = modest_conf_get_bool (priv->modest_conf, keyname, err);
586                 
587         g_free (keyname);
588
589         return retval;
590 }
591
592
593 gboolean
594 modest_account_mgr_set_string (ModestAccountMgr * self, const gchar * name,
595                                const gchar * key, const gchar * val,
596                                gboolean server_account, GError ** err)
597 {
598         ModestAccountMgrPrivate *priv;
599
600         gchar *keyname;
601         gboolean retval;
602
603         g_return_val_if_fail (self, FALSE);
604         g_return_val_if_fail (name, FALSE);
605         g_return_val_if_fail (key, FALSE);
606
607         keyname = get_account_keyname (name, key, server_account);
608         
609         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
610
611         retval = modest_conf_set_string (priv->modest_conf, keyname, val,
612                                          err);
613
614         g_free (keyname);
615         return retval;
616 }
617
618
619 gboolean
620 modest_account_mgr_set_int (ModestAccountMgr * self, const gchar * name,
621                             const gchar * key, int val, gboolean server_account,
622                             GError ** err)
623 {
624         ModestAccountMgrPrivate *priv;
625
626         gchar *keyname;
627         gboolean retval;
628
629         g_return_val_if_fail (self, FALSE);
630         g_return_val_if_fail (name, FALSE);
631         g_return_val_if_fail (key, FALSE);
632
633         keyname = get_account_keyname (name, key, server_account);
634         
635         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
636
637         retval = modest_conf_set_int (priv->modest_conf, keyname, val, err);
638
639         g_free (keyname);
640         return retval;
641 }
642
643
644
645 gboolean
646 modest_account_mgr_set_bool (ModestAccountMgr * self, const gchar * name,
647                              const gchar * key, gboolean val, gboolean server_account, 
648                              GError ** err)
649 {
650         ModestAccountMgrPrivate *priv;
651
652         gchar *keyname;
653         gboolean retval;
654
655         g_return_val_if_fail (self, FALSE);
656         g_return_val_if_fail (name, FALSE);
657         g_return_val_if_fail (key, FALSE);
658
659         keyname = get_account_keyname (name, key, server_account);
660
661         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
662
663         retval = modest_conf_set_bool (priv->modest_conf, keyname, val, err);
664
665         g_free (keyname);
666         return retval;
667 }
668
669
670 gboolean
671 modest_account_mgr_account_exists (ModestAccountMgr * self, const gchar * name,
672                                    gboolean server_account, GError ** err)
673 {
674         ModestAccountMgrPrivate *priv;
675
676         gchar *keyname;
677         gboolean retval;
678
679         g_return_val_if_fail (self, FALSE);
680         g_return_val_if_fail (name, FALSE);
681
682         keyname = get_account_keyname (name, NULL, server_account);
683
684         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
685         retval = modest_conf_key_exists (priv->modest_conf, keyname, err);
686
687         g_free (keyname);
688         return retval;
689 }
690
691
692 /* must be freed by caller */
693 static gchar *
694 get_account_keyname (const gchar * accname, const gchar * name, gboolean server_account)
695 {
696         gchar *namespace, *account_name, *retval;
697         
698         namespace = server_account ? MODEST_SERVER_ACCOUNT_NAMESPACE : MODEST_ACCOUNT_NAMESPACE;
699
700         if (!accname)
701                 return g_strdup (namespace);
702
703         account_name = modest_conf_key_escape (NULL, accname);
704         
705         if (name)
706                 retval = g_strconcat (namespace, "/", accname, "/", name, NULL);
707         else
708                 retval = g_strconcat (namespace, "/", accname, NULL);
709
710         g_free (account_name);
711
712         return retval;
713 }