]> git.llucax.com Git - software/mutt-debian.git/blob - imap/browse.c
Imported Upstream version 1.5.18
[software/mutt-debian.git] / imap / browse.c
1 /*
2  * Copyright (C) 1996-9 Brandon Long <blong@fiction.net>
3  * Copyright (C) 1999-2007 Brendan Cully <brendan@kublai.com>
4  * 
5  *     This program is free software; you can redistribute it and/or modify
6  *     it under the terms of the GNU General Public License as published by
7  *     the Free Software Foundation; either version 2 of the License, or
8  *     (at your option) any later version.
9  * 
10  *     This program is distributed in the hope that it will be useful,
11  *     but WITHOUT ANY WARRANTY; without even the implied warranty of
12  *     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  *     GNU General Public License for more details.
14  * 
15  *     You should have received a copy of the GNU General Public License
16  *     along with this program; if not, write to the Free Software
17  *     Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
18  */ 
19
20 /* Mutt browser support routines */
21
22 #if HAVE_CONFIG_H
23 # include "config.h"
24 #endif
25
26 #include <stdlib.h>
27 #include <ctype.h>
28
29 #include "mutt.h"
30 #include "imap_private.h"
31
32 /* -- forward declarations -- */
33 static int browse_add_list_result (IMAP_DATA* idata, const char* cmd,
34   struct browser_state* state, short isparent);
35 static void imap_add_folder (char delim, char *folder, int noselect,
36   int noinferiors, struct browser_state *state, short isparent);
37 static int compare_names(struct folder_file *a, struct folder_file *b);
38
39 /* imap_browse: IMAP hook into the folder browser, fills out browser_state,
40  *   given a current folder to browse */
41 int imap_browse (char* path, struct browser_state* state)
42 {
43   IMAP_DATA* idata;
44   IMAP_LIST list;
45   char buf[LONG_STRING];
46   char buf2[LONG_STRING];
47   char mbox[LONG_STRING];
48   char list_cmd[5];
49   int n;
50   int nsup;
51   char ctmp;
52   short showparents = 0;
53   int save_lsub;
54   IMAP_MBOX mx;
55
56   if (imap_parse_path (path, &mx))
57   {
58     mutt_error (_("%s is an invalid IMAP path"), path);
59     return -1;
60   }
61
62   save_lsub = option (OPTIMAPCHECKSUBSCRIBED);
63   unset_option (OPTIMAPCHECKSUBSCRIBED);
64   strfcpy (list_cmd, option (OPTIMAPLSUB) ? "LSUB" : "LIST", sizeof (list_cmd));
65
66   if (!(idata = imap_conn_find (&(mx.account), 0)))
67     goto fail;
68
69   mutt_message _("Getting folder list...");
70
71   /* skip check for parents when at the root */
72   if (mx.mbox && mx.mbox[0] != '\0')
73   {
74     int rc;
75     imap_fix_path (idata, mx.mbox, mbox, sizeof (mbox));
76     imap_munge_mbox_name (buf, sizeof (buf), mbox);
77     imap_unquote_string(buf); /* As kludgy as it gets */
78     mbox[sizeof (mbox) - 1] = '\0';
79     strncpy (mbox, buf, sizeof (mbox) - 1);
80     n = mutt_strlen (mbox);
81
82     dprint (3, (debugfile, "imap_browse: mbox: %s\n", mbox));
83
84     /* if our target exists and has inferiors, enter it if we
85      * aren't already going to */
86     if (mbox[n-1] != idata->delim)
87     {
88       snprintf (buf, sizeof (buf), "%s \"\" \"%s\"", list_cmd, mbox);
89       imap_cmd_start (idata, buf);
90       idata->cmdtype = IMAP_CT_LIST;
91       idata->cmddata = &list;
92       do
93       {
94         list.name = 0;
95         rc = imap_cmd_step (idata);
96         if (rc == IMAP_CMD_CONTINUE && list.name)
97         {
98           if (!list.noinferiors && list.name[0] &&
99               !imap_mxcmp (list.name, mbox) &&
100             (n = strlen (mbox)) < sizeof (mbox) - 1)
101           {
102             mbox[n++] = list.delim;
103             mbox[n] = '\0';
104           }
105         }
106       }
107       while (rc == IMAP_CMD_CONTINUE);
108       idata->cmddata = NULL;
109     }
110
111     /* if we're descending a folder, mark it as current in browser_state */
112     if (mbox[n-1] == list.delim)
113     {
114       showparents = 1;
115       imap_qualify_path (buf, sizeof (buf), &mx, mbox);
116       state->folder = safe_strdup (buf);
117       n--;
118     }
119
120     /* Find superiors to list
121      * Note: UW-IMAP servers return folder + delimiter when asked to list
122      *  folder + delimiter. Cyrus servers don't. So we ask for folder,
123      *  and tack on delimiter ourselves.
124      * Further note: UW-IMAP servers return nothing when asked for 
125      *  NAMESPACES without delimiters at the end. Argh! */
126     for (n--; n >= 0 && mbox[n] != list.delim ; n--);
127     if (n > 0)                  /* "aaaa/bbbb/" -> "aaaa" */
128     {
129       /* forget the check, it is too delicate (see above). Have we ever
130        * had the parent not exist? */
131       ctmp = mbox[n];
132       mbox[n] = '\0';
133
134       if (showparents)
135       {
136         dprint (3, (debugfile, "imap_init_browse: adding parent %s\n", mbox));
137         imap_add_folder (list.delim, mbox, 1, 0, state, 1);
138       }
139
140       /* if our target isn't a folder, we are in our superior */
141       if (!state->folder)
142       {
143         /* store folder with delimiter */
144         mbox[n++] = ctmp;
145         ctmp = mbox[n];
146         mbox[n] = '\0';
147         imap_qualify_path (buf, sizeof (buf), &mx, mbox);
148         state->folder = safe_strdup (buf);
149       }
150       mbox[n] = ctmp;
151     } 
152     /* "/bbbb/" -> add  "/", "aaaa/" -> add "" */
153     else
154     {
155       char relpath[2];
156       /* folder may be "/" */
157       snprintf (relpath, sizeof (relpath), "%c" , n < 0 ? '\0' : idata->delim);
158       if (showparents)
159         imap_add_folder (idata->delim, relpath, 1, 0, state, 1); 
160       if (!state->folder)
161       {
162         imap_qualify_path (buf, sizeof (buf), &mx, relpath);
163         state->folder = safe_strdup (buf);
164       }
165     }
166   }
167   else
168     mbox[0] = '\0';
169
170   /* no namespace, no folder: set folder to host only */
171   if (!state->folder)
172   {
173     imap_qualify_path (buf, sizeof (buf), &mx, NULL);
174     state->folder = safe_strdup (buf);
175   }
176
177   nsup = state->entrylen;
178
179   dprint (3, (debugfile, "imap_browse: Quoting mailbox scan: %s -> ", mbox));
180   snprintf (buf, sizeof (buf), "%s%%", mbox);
181   imap_quote_string (buf2, sizeof (buf2), buf);
182   dprint (3, (debugfile, "%s\n", buf2));
183   snprintf (buf, sizeof (buf), "%s \"\" %s", list_cmd, buf2);
184   if (browse_add_list_result (idata, buf, state, 0))
185     goto fail;
186
187   if (!state->entrylen)
188   {
189     mutt_error _("No such folder");
190     goto fail;
191   }
192
193   mutt_clear_error ();
194
195   qsort(&(state->entry[nsup]),state->entrylen-nsup,sizeof(state->entry[0]),
196         (int (*)(const void*,const void*)) compare_names);
197
198   if (save_lsub)
199     set_option (OPTIMAPCHECKSUBSCRIBED);
200
201   FREE (&mx.mbox);
202   return 0;
203
204  fail:
205   if (save_lsub)
206     set_option (OPTIMAPCHECKSUBSCRIBED);
207   FREE (&mx.mbox);
208   return -1;
209 }
210
211 int imap_mailbox_state (const char* path, struct mailbox_state* state)
212 {
213   IMAP_DATA* idata;
214   IMAP_MBOX mx;
215   IMAP_STATUS* status;
216
217   memset (state, 0, sizeof (*state));
218   if (imap_parse_path (path, &mx) < 0)
219   {
220     dprint (1, (debugfile, "imap_mailbox_state: bad path %s\n", path));
221     return -1;
222   }
223   if (!(idata = imap_conn_find (&mx.account, M_IMAP_CONN_NONEW)))
224   {
225     dprint (2, (debugfile, "imap_mailbox_state: no open connection for %s\n",
226                 path));
227     FREE (&mx.mbox);
228     return -1;
229   }
230
231   if (!imap_mxcmp(mx.mbox, idata->mailbox))
232   {
233     state->new = idata->ctx->new;
234     state->messages = idata->ctx->msgcount;
235   }
236   else if ((status = imap_mboxcache_get (idata, mx.mbox, 0)))
237   {
238     state->new = status->unseen;
239     state->messages = status->messages;
240   }
241
242   return 0;
243 }
244
245 /* imap_mailbox_create: Prompt for a new mailbox name, and try to create it */
246 int imap_mailbox_create (const char* folder)
247 {
248   IMAP_DATA* idata;
249   IMAP_MBOX mx;
250   char buf[LONG_STRING];
251   short n;
252
253   if (imap_parse_path (folder, &mx) < 0)
254   {
255     dprint (1, (debugfile, "imap_mailbox_create: Bad starting path %s\n",
256       folder));
257     return -1;
258   }
259
260   if (!(idata = imap_conn_find (&mx.account, M_IMAP_CONN_NONEW)))
261   {
262     dprint (1, (debugfile, "imap_mailbox_create: Couldn't find open connection to %s", mx.account.host));
263     goto fail;
264   }
265   
266   strfcpy (buf, NONULL (mx.mbox), sizeof (buf));
267
268   /* append a delimiter if necessary */
269   n = mutt_strlen (buf);
270   if (n && (n < sizeof (buf) - 1) && (buf[n-1] != idata->delim))
271   {
272     buf[n++] = idata->delim;
273     buf[n] = '\0';
274   }
275   
276   if (mutt_get_field (_("Create mailbox: "), buf, sizeof (buf), M_FILE) < 0)
277     goto fail;
278
279   if (!mutt_strlen (buf))
280   {
281     mutt_error (_("Mailbox must have a name."));
282     mutt_sleep(1);
283     goto fail;
284   }
285   
286   if (imap_create_mailbox (idata, buf) < 0)
287     goto fail;
288
289   mutt_message _("Mailbox created.");
290   mutt_sleep (0);
291
292   FREE (&mx.mbox);
293   return 0;
294
295  fail:
296   FREE (&mx.mbox);
297   return -1;
298 }
299
300 int imap_mailbox_rename(const char* mailbox)
301 {
302   IMAP_DATA* idata;
303   IMAP_MBOX mx;
304   char buf[LONG_STRING];
305   char newname[SHORT_STRING];
306
307   if (imap_parse_path (mailbox, &mx) < 0)
308   {
309     dprint (1, (debugfile, "imap_mailbox_rename: Bad source mailbox %s\n",
310       mailbox));
311     return -1;
312   }
313
314   if (!(idata = imap_conn_find (&mx.account, M_IMAP_CONN_NONEW)))
315   {
316     dprint (1, (debugfile, "imap_mailbox_rename: Couldn't find open connection to %s", mx.account.host));
317     goto fail;
318   }
319
320   snprintf(buf, sizeof (buf), _("Rename mailbox %s to: "), mx.mbox);
321   
322  if (mutt_get_field (buf, newname, sizeof (newname), M_FILE) < 0)
323     goto fail;
324
325   if (!mutt_strlen (newname))
326   {
327     mutt_error (_("Mailbox must have a name."));
328     mutt_sleep (1);
329     goto fail;
330   }
331
332   if (imap_rename_mailbox (idata, &mx, newname) < 0) {
333     mutt_error (_("Rename failed: %s"), imap_get_qualifier (idata->buf));
334     mutt_sleep (1);
335     goto fail;
336   }
337
338   mutt_message (_("Mailbox renamed."));
339   mutt_sleep (0);
340
341   FREE (&mx.mbox);
342   return 0;
343
344  fail:
345   FREE (&mx.mbox);
346   return -1;
347 }
348
349 static int browse_add_list_result (IMAP_DATA* idata, const char* cmd,
350   struct browser_state* state, short isparent)
351 {
352   IMAP_LIST list;
353   IMAP_MBOX mx;
354   int rc;
355
356   if (imap_parse_path (state->folder, &mx))
357   {
358     dprint (2, (debugfile,
359       "browse_add_list_result: current folder %s makes no sense\n", state->folder));
360     return -1;
361   }
362
363   imap_cmd_start (idata, cmd);
364   idata->cmdtype = IMAP_CT_LIST;
365   idata->cmddata = &list;
366   do
367   {
368     list.name = NULL;
369     rc = imap_cmd_step (idata);
370
371     if (rc == IMAP_CMD_CONTINUE && list.name)
372     {
373       /* Let a parent folder never be selectable for navigation */
374       if (isparent)
375         list.noselect = 1;
376       /* prune current folder from output */
377       if (isparent || mutt_strncmp (list.name, mx.mbox, strlen (list.name)))
378         imap_add_folder (list.delim, list.name, list.noselect, list.noinferiors,
379                          state, isparent);
380     }
381   }
382   while (rc == IMAP_CMD_CONTINUE);
383   idata->cmddata = NULL;
384
385   FREE (&mx.mbox);
386   return rc == IMAP_CMD_OK ? 0 : -1;
387 }
388
389 /* imap_add_folder: add a folder name to the browser list, formatting it as
390  *   necessary. */
391 static void imap_add_folder (char delim, char *folder, int noselect,
392   int noinferiors, struct browser_state *state, short isparent)
393 {
394   char tmp[LONG_STRING];
395   char relpath[LONG_STRING];
396   IMAP_MBOX mx;
397
398   if (imap_parse_path (state->folder, &mx))
399     return;
400
401   imap_unmunge_mbox_name (folder);
402
403   if (state->entrylen + 1 == state->entrymax)
404   {
405     safe_realloc (&state->entry,
406       sizeof (struct folder_file) * (state->entrymax += 256));
407     memset (state->entry + state->entrylen, 0,
408       (sizeof (struct folder_file) * (state->entrymax - state->entrylen)));
409   }
410
411   /* render superiors as unix-standard ".." */
412   if (isparent)
413     strfcpy (relpath, "../", sizeof (relpath));
414   /* strip current folder from target, to render a relative path */
415   else if (!mutt_strncmp (mx.mbox, folder, mutt_strlen (mx.mbox)))
416     strfcpy (relpath, folder + mutt_strlen (mx.mbox), sizeof (relpath));
417   else
418     strfcpy (relpath, folder, sizeof (relpath));
419
420   /* apply filemask filter. This should really be done at menu setup rather
421    * than at scan, since it's so expensive to scan. But that's big changes
422    * to browser.c */
423   if (!((regexec (Mask.rx, relpath, 0, NULL, 0) == 0) ^ Mask.not))
424   {
425     FREE (&mx.mbox);
426     return;
427   }
428
429   imap_qualify_path (tmp, sizeof (tmp), &mx, folder);
430   (state->entry)[state->entrylen].name = safe_strdup (tmp);
431
432   /* mark desc with delim in browser if it can have subfolders */
433   if (!isparent && !noinferiors && strlen (relpath) < sizeof (relpath) - 1)
434   {
435     relpath[strlen (relpath) + 1] = '\0';
436     relpath[strlen (relpath)] = delim;
437   }
438   
439   (state->entry)[state->entrylen].desc = safe_strdup (relpath);
440
441   (state->entry)[state->entrylen].imap = 1;
442   /* delimiter at the root is useless. */
443   if (folder[0] == '\0')
444     delim = '\0';
445   (state->entry)[state->entrylen].delim = delim;
446   (state->entry)[state->entrylen].selectable = !noselect;
447   (state->entry)[state->entrylen].inferiors = !noinferiors;
448   (state->entrylen)++;
449
450   FREE (&mx.mbox);
451 }
452
453 static int compare_names(struct folder_file *a, struct folder_file *b) 
454 {
455   return mutt_strcmp(a->name, b->name);
456 }