Rearranged whitespace (and comments) in all the code!
[mmh] / uip / packf.c
1 /*
2  * packf.c -- pack a nmh folder into a file
3  *
4  * This code is Copyright (c) 2002, by the authors of nmh.  See the
5  * COPYRIGHT file in the root directory of the nmh distribution for
6  * complete copyright information.
7  */
8
9 #include <h/mh.h>
10 #include <fcntl.h>
11 #include <h/dropsbr.h>
12 #include <h/utils.h>
13 #include <errno.h>
14
15 static struct swit switches[] = {
16 #define FILESW  0
17         { "file name", 0 },
18 #define MBOXSW  1
19         { "mbox", 0 },
20 #define MMDFSW  2
21         { "mmdf", 0 },
22 #define VERSIONSW  3
23         { "version", 0 },
24 #define HELPSW  4
25         { "help", 0 },
26         { NULL, 0 }
27 };
28
29 static int md = NOTOK;
30 static int mbx_style = MBOX_FORMAT;
31 static int mapping = 0;
32
33 static void mbxclose_done(int) NORETURN;
34
35 char *file = NULL;
36
37
38 int
39 main (int argc, char **argv)
40 {
41         int fd, msgnum;
42         char *cp, *maildir, *msgnam, *folder = NULL, buf[BUFSIZ];
43         char **argp, **arguments;
44         struct msgs_array msgs = { 0, 0, NULL };
45         struct msgs *mp;
46         struct stat st;
47
48         done=mbxclose_done;
49
50 #ifdef LOCALE
51         setlocale(LC_ALL, "");
52 #endif
53         invo_name = r1bindex (argv[0], '/');
54
55         /* read user profile/context */
56         context_read();
57
58         arguments = getarguments (invo_name, argc, argv, 1);
59         argp = arguments;
60
61         /*
62          * Parse arguments
63          */
64         while ((cp = *argp++)) {
65                 if (*cp == '-') {
66                         switch (smatch (++cp, switches)) {
67                                 case AMBIGSW:
68                                         ambigsw (cp, switches);
69                                         done (1);
70                                 case UNKWNSW:
71                                         adios (NULL, "-%s unknown", cp);
72
73                                 case HELPSW:
74                                         snprintf (buf, sizeof(buf), "%s [+folder] [msgs] [switches]",
75                                                 invo_name);
76                                         print_help (buf, switches, 1);
77                                         done (1);
78                                 case VERSIONSW:
79                                         print_version(invo_name);
80                                         done (1);
81
82                                 case FILESW:
83                                         if (file)
84                                                 adios (NULL, "only one file at a time!");
85                                         if (!(file = *argp++) || *file == '-')
86                                                 adios (NULL, "missing argument to %s", argp[-2]);
87                                         continue;
88
89                                 case MBOXSW:
90                                         mbx_style = MBOX_FORMAT;
91                                         mapping = 0;
92                                         continue;
93                                 case MMDFSW:
94                                         mbx_style = MMDF_FORMAT;
95                                         mapping = 1;
96                                         continue;
97                         }
98                 }
99                 if (*cp == '+' || *cp == '@') {
100                         if (folder)
101                                 adios (NULL, "only one folder at a time!");
102                         folder = pluspath (cp);
103                 } else
104                                 app_msgarg(&msgs, cp);
105         }
106
107         if (!file)
108                 file = "./msgbox";
109         file = path (file, TFILE);
110
111         /*
112          * Check if file to be created (or appended to)
113          * exists.  If not, ask for confirmation.
114          */
115         if (stat (file, &st) == NOTOK) {
116                 if (errno != ENOENT)
117                         adios (file, "error on file");
118                 cp = concat ("Create file \"", file, "\"? ", NULL);
119                 if (!getanswer (cp))
120                         done (1);
121                 free (cp);
122         }
123
124         if (!context_find ("path"))
125                 free (path ("./", TFOLDER));
126
127         /* default is to pack whole folder */
128         if (!msgs.size)
129                 app_msgarg(&msgs, "all");
130
131         if (!folder)
132                 folder = getfolder (1);
133         maildir = m_maildir (folder);
134
135         if (chdir (maildir) == NOTOK)
136                 adios (maildir, "unable to change directory to ");
137
138         /* read folder and create message structure */
139         if (!(mp = folder_read (folder)))
140                 adios (NULL, "unable to read folder %s", folder);
141
142         /* check for empty folder */
143         if (mp->nummsg == 0)
144                 adios (NULL, "no messages in %s", folder);
145
146         /* parse all the message ranges/sequences and set SELECTED */
147         for (msgnum = 0; msgnum < msgs.size; msgnum++)
148                 if (!m_convert (mp, msgs.msgs[msgnum]))
149                         done (1);
150         seq_setprev (mp);  /* set the previous-sequence */
151
152         /* open and lock new maildrop file */
153         if ((md = mbx_open(file, mbx_style, getuid(), getgid(), m_gmprot())) == NOTOK)
154                 adios (file, "unable to open");
155
156         /* copy all the SELECTED messages to the file */
157         for (msgnum = mp->lowsel; msgnum <= mp->hghsel; msgnum++)
158                 if (is_selected(mp, msgnum)) {
159                         if ((fd = open (msgnam = m_name (msgnum), O_RDONLY)) == NOTOK) {
160                                 admonish (msgnam, "unable to read message");
161                                 break;
162                         }
163
164                         if (mbx_copy (file, mbx_style, md, fd, mapping, NULL, 1) == NOTOK)
165                                 adios (file, "error writing to file");
166
167                         close (fd);
168                 }
169
170         /* close and unlock maildrop file */
171         mbx_close (file, md);
172
173         context_replace (pfolder, folder);  /* update current folder */
174         if (mp->hghsel != mp->curmsg)
175                 seq_setcur (mp, mp->lowsel);
176         seq_save (mp);
177         context_save ();  /* save the context file */
178         folder_free (mp);  /* free folder/message structure */
179         done (0);
180         return 1;
181 }
182
183 static void
184 mbxclose_done (int status)
185 {
186         mbx_close (file, md);
187         exit (status);
188 }