* improvements in display of address, refactored some code
[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_is_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         priv->modest_conf = NULL;
221 }
222
223
224 ModestAccountMgr *
225 modest_account_mgr_new (ModestConf * conf)
226 {
227         GObject *obj;
228         ModestAccountMgrPrivate *priv;
229
230         g_return_val_if_fail (conf, NULL);
231
232         obj = G_OBJECT (g_object_new (MODEST_TYPE_ACCOUNT_MGR, NULL));
233         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (obj);
234
235         priv->modest_conf      = conf;
236
237         g_signal_connect (G_OBJECT (conf), "key_changed",
238                           G_CALLBACK (on_key_change),
239                           obj);
240         
241         return MODEST_ACCOUNT_MGR (obj);
242 }
243
244
245 static const gchar *
246 null_means_empty (const gchar * str)
247 {
248         return str ? str : "";
249 }
250
251
252
253 gboolean
254 modest_account_mgr_disable_account (ModestAccountMgr *self, const gchar* name,
255                                     gboolean is_server_account)
256 {
257         return modest_account_mgr_set_bool (self, name,
258                                             MODEST_ACCOUNT_ENABLED, FALSE,
259                                             is_server_account, NULL);
260 }
261
262
263 gboolean
264 modest_account_mgr_enable_account (ModestAccountMgr *self, const gchar* name,
265                                    gboolean is_server_account)
266 {
267         return modest_account_mgr_set_bool (self, name,
268                                             MODEST_ACCOUNT_ENABLED, TRUE,
269                                             is_server_account, NULL);
270 }
271
272
273 gboolean
274 modest_account_mgr_account_is_enabled (ModestAccountMgr *self, const gchar* name,
275                                        gboolean is_server_account)
276 {
277         return modest_account_mgr_get_bool (self, name,
278                                             MODEST_ACCOUNT_ENABLED, is_server_account,
279                                             NULL);
280 }
281
282
283 gboolean
284 modest_account_mgr_add_account (ModestAccountMgr *self,
285                                 const gchar *name,
286                                 const gchar *store_account,
287                                 const gchar *transport_account,
288                                 GError **err)
289 {
290         ModestAccountMgrPrivate *priv;
291         gchar *key;
292         gboolean ok;
293
294         g_return_val_if_fail (self, FALSE);
295         g_return_val_if_fail (name, FALSE);
296
297         if (modest_account_mgr_account_exists (self, name, FALSE, err)) {
298                 g_printerr ("modest: account already exists\n");
299                 return FALSE;
300         }
301         
302         /*
303          * we create the account by adding an account 'dir', with the name <name>,
304          * and in that the 'display_name' string key
305          */
306         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
307         
308         key = get_account_keyname (name, MODEST_ACCOUNT_DISPLAY_NAME, FALSE);
309         ok = modest_conf_set_string (priv->modest_conf, key, name, err);
310         g_free (key);
311
312         if (!ok) {
313                 g_printerr ("modest: cannot set display name\n");
314                 return FALSE;
315         }
316         
317         if (store_account) {
318                 key = get_account_keyname (name, MODEST_ACCOUNT_STORE_ACCOUNT, FALSE);
319                 ok = modest_conf_set_string (priv->modest_conf, key, store_account, err);
320                 g_free (key);
321                 if (!ok) {
322                         g_printerr ("modest: failed to set store account '%s'\n",
323                                 store_account);
324                         return FALSE;
325                 }
326         }
327
328         if (transport_account) {
329                 key = get_account_keyname (name, MODEST_ACCOUNT_TRANSPORT_ACCOUNT, FALSE);
330                 ok = modest_conf_set_string (priv->modest_conf, key, transport_account, err);
331                 g_free (key);
332                 if (!ok) {
333                         g_printerr ("modest: failed to set transport account '%s'\n",
334                                 transport_account);
335                         return FALSE;
336                 }
337         }
338
339         modest_account_mgr_enable_account (self, name, FALSE);
340         
341         return TRUE;
342 }
343
344
345
346
347 gboolean
348 modest_account_mgr_add_server_account (ModestAccountMgr * self,
349                                        const gchar * name,
350                                        const gchar * hostname,
351                                        const gchar * username,
352                                        const gchar * password,
353                                        const gchar * proto)
354 {
355         ModestAccountMgrPrivate *priv;
356         gchar *key;
357
358         g_return_val_if_fail (self, FALSE);
359         g_return_val_if_fail (name, FALSE);
360
361         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
362
363         key = get_account_keyname (name, NULL, TRUE);
364         if (modest_conf_key_exists (priv->modest_conf, key, NULL)) {
365                 g_printerr ("modest: server account '%s' already exists", name);
366                 g_free (key);
367                 return FALSE;
368         }
369         g_free (key);
370         
371         /* hostname */
372         key = get_account_keyname (name, MODEST_ACCOUNT_HOSTNAME, TRUE);
373         modest_conf_set_string (priv->modest_conf, key, null_means_empty(hostname), NULL);
374         g_free (key);
375
376         /* username */
377         key = get_account_keyname (name, MODEST_ACCOUNT_USERNAME, TRUE);
378         modest_conf_set_string (priv->modest_conf, key, null_means_empty (username), NULL);
379         g_free (key);
380
381         /* password */
382         key = get_account_keyname (name, MODEST_ACCOUNT_PASSWORD, TRUE);
383         modest_conf_set_string (priv->modest_conf, key, null_means_empty (password), NULL);
384         g_free (key);
385
386         /* proto */
387         key = get_account_keyname (name, MODEST_ACCOUNT_PROTO, TRUE);
388         modest_conf_set_string (priv->modest_conf, key, null_means_empty (proto), NULL);
389         g_free (key);
390
391         /* enable it */
392         modest_account_mgr_enable_account (self, name, TRUE);
393         
394         return TRUE;
395 }
396
397
398
399 gboolean
400 modest_account_mgr_remove_account (ModestAccountMgr * self,
401                                    const gchar * name,
402                                    gboolean server_account,
403                                    GError ** err)
404 {
405         ModestAccountMgrPrivate *priv;
406         gchar *key;
407         gboolean retval;
408
409         g_return_val_if_fail (self, FALSE);
410         g_return_val_if_fail (name, FALSE);
411
412         if (!modest_account_mgr_account_exists (self, name, server_account, err)) {
413                 g_printerr ("modest: account '%s' does not exist\n", name);
414                 return FALSE;
415         }
416
417         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
418         key = get_account_keyname (name, NULL, server_account);
419
420         retval = modest_conf_remove_key (priv->modest_conf, key, NULL);
421
422         g_free (key);
423         return retval;
424 }
425
426
427
428 /* strip the first /n/ character from each element */
429 /* caller must make sure all elements are strings with
430  * length >= n, and also that data can be freed.
431  */
432 static GSList*
433 strip_prefix_from_elements (GSList * lst, guint n)
434 {
435         GSList *cursor = lst;
436
437         while (cursor) {
438                 gchar *str = (gchar *) cursor->data;
439                 cursor->data = g_strdup (str + n);
440                 g_free (str);
441                 cursor = cursor->next;
442         }
443         return lst;
444 }
445
446
447 GSList *
448 modest_account_mgr_search_server_accounts (ModestAccountMgr * self,
449                                            const gchar * account_name,
450                                            ModestProtoType type,
451                                            const gchar *proto,
452                                            gboolean only_enabled)
453 {
454         GSList *accounts;
455         GSList *cursor;
456         ModestAccountMgrPrivate *priv;
457         gchar *key;
458         GError *err = NULL;
459         
460         g_return_val_if_fail (self, NULL);
461         
462         key      = get_account_keyname (account_name, NULL, TRUE);
463         priv     = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
464         
465         /* get the list of all server accounts */
466         accounts = modest_conf_list_subkeys (priv->modest_conf, key, &err);
467         if (err) {
468                 g_error_free (err);
469                 g_printerr ("modest: failed to get subkeys for '%s'\n", key);
470                 return NULL;
471         }
472         
473         /* no restrictions, return everything */
474         if (type == MODEST_PROTO_TYPE_ANY && !proto)
475                 return strip_prefix_from_elements (accounts, strlen(key)+1);
476         /* +1 because we must remove the ending '/' as well */
477         
478         /* otherwise, filter out the none-matching ones */
479         cursor = accounts;
480         while (cursor) {
481                 gchar *account;
482                 gchar *acc_proto;
483                 gboolean enabled;
484                 
485                 account = account_from_key ((gchar*)cursor->data, NULL, NULL);
486                 
487                 enabled   = modest_account_mgr_account_is_enabled (self, account, TRUE);
488                 acc_proto = modest_account_mgr_get_string (self, account, MODEST_ACCOUNT_PROTO,
489                                                            TRUE, NULL);
490                 if ((!acc_proto) ||                                /* proto not defined? */
491                     (type != MODEST_PROTO_TYPE_ANY &&              /* proto type ...     */
492                      modest_proto_type (acc_proto) != type) ||     /* ... matches?       */
493                     (proto && strcmp (proto, acc_proto) != 0) ||   /* proto matches?     */
494                     (!enabled && only_enabled)) {                  /* account enabled?   */
495                         /* match! remove from the list */
496                         GSList *nxt = cursor->next;
497                         accounts = g_slist_delete_link (accounts, cursor);
498                         cursor = nxt;
499                 } else
500                         cursor = cursor->next;
501
502                 g_free (account);
503                 g_free (acc_proto);
504         }
505
506         return strip_prefix_from_elements (accounts, strlen(key)+1);
507         /* +1 because we must remove the ending '/' as well */
508 }
509
510
511 GSList *
512 modest_account_mgr_account_names (ModestAccountMgr * self, GError ** err)
513 {
514         GSList *accounts;
515         ModestAccountMgrPrivate *priv;
516         const size_t prefix_len = strlen (MODEST_ACCOUNT_NAMESPACE "/");
517
518
519         g_return_val_if_fail (self, NULL);
520
521         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
522
523         accounts = modest_conf_list_subkeys (priv->modest_conf,
524                                              MODEST_ACCOUNT_NAMESPACE, err);
525         return strip_prefix_from_elements (accounts, prefix_len);
526 }
527
528
529 gchar *
530 modest_account_mgr_get_string (ModestAccountMgr *self, const gchar *name,
531                                const gchar *key, gboolean server_account, GError **err) {
532
533         ModestAccountMgrPrivate *priv;
534
535         gchar *keyname;
536         gchar *retval;
537
538         g_return_val_if_fail (self, NULL);
539         g_return_val_if_fail (name, NULL);
540         g_return_val_if_fail (key, NULL);
541
542         keyname = get_account_keyname (name, key, server_account);
543         
544         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
545         retval = modest_conf_get_string (priv->modest_conf, keyname, err);
546         g_free (keyname);
547
548         return retval;
549 }
550
551
552 gint
553 modest_account_mgr_get_int (ModestAccountMgr *self, const gchar *name,
554                             const gchar *key, gboolean server_account, GError **err)
555 {
556         ModestAccountMgrPrivate *priv;
557
558         gchar *keyname;
559         gint retval;
560
561         g_return_val_if_fail (self, -1);
562         g_return_val_if_fail (name, -1);
563         g_return_val_if_fail (key, -1);
564
565         keyname = get_account_keyname (name, key, server_account);
566
567         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
568         retval = modest_conf_get_int (priv->modest_conf, keyname, err);
569         g_free (keyname);
570
571         return retval;
572 }
573
574
575
576 gboolean
577 modest_account_mgr_get_bool (ModestAccountMgr * self, const gchar *account,
578                              const gchar * key, gboolean server_account, GError ** err)
579 {
580         ModestAccountMgrPrivate *priv;
581
582         gchar *keyname;
583         gboolean retval;
584
585         g_return_val_if_fail (self, FALSE);
586         g_return_val_if_fail (account, FALSE);
587         g_return_val_if_fail (key, FALSE);
588
589         keyname = get_account_keyname (account, key, server_account);
590         
591         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
592         retval = modest_conf_get_bool (priv->modest_conf, keyname, err);
593                 
594         g_free (keyname);
595
596         return retval;
597 }
598
599
600 gboolean
601 modest_account_mgr_set_string (ModestAccountMgr * self, const gchar * name,
602                                const gchar * key, const gchar * val,
603                                gboolean server_account, GError ** err)
604 {
605         ModestAccountMgrPrivate *priv;
606
607         gchar *keyname;
608         gboolean retval;
609
610         g_return_val_if_fail (self, FALSE);
611         g_return_val_if_fail (name, FALSE);
612         g_return_val_if_fail (key, FALSE);
613
614         keyname = get_account_keyname (name, key, server_account);
615         
616         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
617
618         retval = modest_conf_set_string (priv->modest_conf, keyname, val,
619                                          err);
620
621         g_free (keyname);
622         return retval;
623 }
624
625
626 gboolean
627 modest_account_mgr_set_int (ModestAccountMgr * self, const gchar * name,
628                             const gchar * key, int val, gboolean server_account,
629                             GError ** err)
630 {
631         ModestAccountMgrPrivate *priv;
632
633         gchar *keyname;
634         gboolean retval;
635
636         g_return_val_if_fail (self, FALSE);
637         g_return_val_if_fail (name, FALSE);
638         g_return_val_if_fail (key, FALSE);
639
640         keyname = get_account_keyname (name, key, server_account);
641         
642         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
643
644         retval = modest_conf_set_int (priv->modest_conf, keyname, val, err);
645
646         g_free (keyname);
647         return retval;
648 }
649
650
651
652 gboolean
653 modest_account_mgr_set_bool (ModestAccountMgr * self, const gchar * name,
654                              const gchar * key, gboolean val, gboolean server_account, 
655                              GError ** err)
656 {
657         ModestAccountMgrPrivate *priv;
658
659         gchar *keyname;
660         gboolean retval;
661
662         g_return_val_if_fail (self, FALSE);
663         g_return_val_if_fail (name, FALSE);
664         g_return_val_if_fail (key, FALSE);
665
666         keyname = get_account_keyname (name, key, server_account);
667
668         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
669
670         retval = modest_conf_set_bool (priv->modest_conf, keyname, val, err);
671
672         g_free (keyname);
673         return retval;
674 }
675
676
677 gboolean
678 modest_account_mgr_account_exists (ModestAccountMgr * self, const gchar * name,
679                                    gboolean server_account, GError ** err)
680 {
681         ModestAccountMgrPrivate *priv;
682
683         gchar *keyname;
684         gboolean retval;
685
686         g_return_val_if_fail (self, FALSE);
687         g_return_val_if_fail (name, FALSE);
688
689         keyname = get_account_keyname (name, NULL, server_account);
690
691         priv = MODEST_ACCOUNT_MGR_GET_PRIVATE (self);
692         retval = modest_conf_key_exists (priv->modest_conf, keyname, err);
693
694         g_free (keyname);
695         return retval;
696 }
697
698
699 /* must be freed by caller */
700 static gchar *
701 get_account_keyname (const gchar * accname, const gchar * name, gboolean server_account)
702 {
703         gchar *namespace, *account_name, *retval;
704         
705         namespace = server_account ? MODEST_SERVER_ACCOUNT_NAMESPACE : MODEST_ACCOUNT_NAMESPACE;
706
707         if (!accname)
708                 return g_strdup (namespace);
709
710         account_name = modest_conf_key_escape (NULL, accname);
711         
712         if (name)
713                 retval = g_strconcat (namespace, "/", accname, "/", name, NULL);
714         else
715                 retval = g_strconcat (namespace, "/", accname, NULL);
716
717         g_free (account_name);
718
719         return retval;
720 }