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