0
|
1 /* pick.c - select messages by content */
|
|
2 #ifndef lint
|
|
3 static char ident[] = "@(#)$Id$";
|
|
4 #endif /* lint */
|
|
5
|
|
6 #include "../h/mh.h"
|
|
7 #include "../zotnet/tws.h"
|
|
8 #include <stdio.h>
|
|
9 #ifdef LOCALE
|
|
10 #include <locale.h>
|
|
11 #endif
|
|
12
|
|
13 /* */
|
|
14
|
|
15 static struct swit switches[] = {
|
|
16 #define ANDSW 0
|
|
17 "and", 0,
|
|
18 #define ORSW 1
|
|
19 "or", 0,
|
|
20 #define NOTSW 2
|
|
21 "not", 0,
|
|
22 #define LBRSW 3
|
|
23 "lbrace", 0,
|
|
24 #define RBRSW 4
|
|
25 "rbrace", 0,
|
|
26
|
|
27 #define CCSW 5
|
|
28 "cc pattern", 0,
|
|
29 #define DATESW 6
|
|
30 "date pattern", 0,
|
|
31 #define FROMSW 7
|
|
32 "from pattern", 0,
|
|
33 #define SRCHSW 8
|
|
34 "search pattern", 0,
|
|
35 #define SUBJSW 9
|
|
36 "subject pattern", 0,
|
|
37 #define TOSW 10
|
|
38 "to pattern", 0,
|
|
39 #define OTHRSW 11
|
|
40 "-othercomponent pattern", 0,
|
|
41 #define AFTRSW 12
|
|
42 "after date", 0,
|
|
43 #define BEFRSW 13
|
|
44 "before date", 0,
|
|
45 #define DATFDSW 14
|
|
46 "datefield field", 5,
|
|
47
|
|
48 #define SEQSW 15
|
|
49 "sequence name", 0,
|
|
50 #define PUBLSW 16
|
|
51 "public", 0,
|
|
52 #define NPUBLSW 17
|
|
53 "nopublic", 0,
|
|
54 #define ZEROSW 18
|
|
55 "zero", 0,
|
|
56 #define NZEROSW 19
|
|
57 "nozero", 0,
|
|
58
|
|
59 #define LISTSW 20
|
|
60 "list", 0,
|
|
61 #define NLISTSW 21
|
|
62 "nolist", 0,
|
|
63
|
|
64 #define HELPSW 22
|
|
65 "help", 4,
|
|
66
|
|
67 NULL, 0
|
|
68 };
|
|
69
|
|
70 /* */
|
|
71
|
|
72 static int listsw = 0;
|
|
73
|
|
74 /* */
|
|
75
|
|
76 /* ARGSUSED */
|
|
77
|
|
78 main (argc, argv)
|
|
79 char *argv[];
|
|
80 {
|
|
81 int publicsw = -1,
|
|
82 zerosw = 1,
|
|
83 msgp = 0,
|
|
84 seqp = 0,
|
|
85 vecp = 0,
|
|
86 lo,
|
|
87 hi,
|
|
88 msgnum;
|
|
89 char *maildir,
|
|
90 *folder = NULL,
|
|
91 buf[100],
|
|
92 *cp,
|
|
93 **ap,
|
|
94 **argp,
|
|
95 *arguments[MAXARGS],
|
|
96 *msgs[MAXARGS],
|
|
97 *seqs[NATTRS + 1],
|
|
98 *vec[MAXARGS];
|
|
99 struct msgs *mp;
|
|
100 register FILE *fp;
|
|
101
|
|
102 #ifdef LOCALE
|
|
103 setlocale(LC_ALL, "");
|
|
104 #endif
|
|
105 #ifdef JAPAN
|
|
106 ml_init();
|
|
107 #endif /* JAPAN */
|
|
108 invo_name = r1bindex (argv[0], '/');
|
|
109 if ((cp = m_find (invo_name)) != NULL) {
|
|
110 ap = brkstring (cp = getcpy (cp), " ", "\n");
|
|
111 ap = copyip (ap, arguments);
|
|
112 }
|
|
113 else
|
|
114 ap = arguments;
|
|
115 (void) copyip (argv + 1, ap);
|
|
116 argp = arguments;
|
|
117
|
|
118 /* */
|
|
119
|
|
120 while (cp = *argp++) {
|
|
121 if (*cp == '-') {
|
|
122 if (*++cp == '-') {
|
|
123 vec[vecp++] = --cp;
|
|
124 goto pattern;
|
|
125 }
|
|
126 switch (smatch (cp, switches)) {
|
|
127 case AMBIGSW:
|
|
128 ambigsw (cp, switches);
|
|
129 done (1);
|
|
130 case UNKWNSW:
|
|
131 adios (NULLCP, "-%s unknown", cp);
|
|
132 case HELPSW:
|
|
133 (void) sprintf (buf, "%s [+folder] [msgs] [switches]",
|
|
134 invo_name);
|
|
135 help (buf, switches);
|
|
136 listsw = 0; /* HACK */
|
|
137 done (1);
|
|
138
|
|
139 case CCSW:
|
|
140 case DATESW:
|
|
141 case FROMSW:
|
|
142 case SUBJSW:
|
|
143 case TOSW:
|
|
144 case DATFDSW:
|
|
145 case AFTRSW:
|
|
146 case BEFRSW:
|
|
147 case SRCHSW:
|
|
148 vec[vecp++] = --cp;
|
|
149 pattern: ;
|
|
150 if (!(cp = *argp++))/* allow -xyz arguments */
|
|
151 adios (NULLCP, "missing argument to %s", argp[-2]);
|
|
152 vec[vecp++] = cp;
|
|
153 continue;
|
|
154 case OTHRSW:
|
|
155 adios (NULLCP, "internal error!");
|
|
156
|
|
157 case ANDSW:
|
|
158 case ORSW:
|
|
159 case NOTSW:
|
|
160 case LBRSW:
|
|
161 case RBRSW:
|
|
162 vec[vecp++] = --cp;
|
|
163 continue;
|
|
164
|
|
165 case SEQSW:
|
|
166 if (!(cp = *argp++) || *cp == '-')
|
|
167 adios (NULLCP, "missing argument to %s", argp[-2]);
|
|
168 if (seqp < NATTRS)
|
|
169 seqs[seqp++] = cp;
|
|
170 else
|
|
171 adios (NULLCP, "only %d sequences allowed!", NATTRS);
|
|
172 listsw = 0;
|
|
173 continue;
|
|
174 case PUBLSW:
|
|
175 publicsw = 1;
|
|
176 continue;
|
|
177 case NPUBLSW:
|
|
178 publicsw = 0;
|
|
179 continue;
|
|
180 case ZEROSW:
|
|
181 zerosw++;
|
|
182 continue;
|
|
183 case NZEROSW:
|
|
184 zerosw = 0;
|
|
185 continue;
|
|
186
|
|
187 case LISTSW:
|
|
188 listsw++;
|
|
189 continue;
|
|
190 case NLISTSW:
|
|
191 listsw = 0;
|
|
192 continue;
|
|
193 }
|
|
194 }
|
|
195 if (*cp == '+' || *cp == '@')
|
|
196 if (folder)
|
|
197 adios (NULLCP, "only one folder at a time!");
|
|
198 else
|
|
199 folder = path (cp + 1, *cp == '+' ? TFOLDER : TSUBCWF);
|
|
200 else
|
|
201 msgs[msgp++] = cp;
|
|
202 }
|
|
203 vec[vecp] = NULL;
|
|
204
|
|
205 /* */
|
|
206
|
|
207 if (!m_find ("path"))
|
|
208 free (path ("./", TFOLDER));
|
|
209 if (!msgp)
|
|
210 msgs[msgp++] = "all";
|
|
211 if (!folder)
|
|
212 folder = m_getfolder ();
|
|
213 maildir = m_maildir (folder);
|
|
214
|
|
215 if (chdir (maildir) == NOTOK)
|
|
216 adios (maildir, "unable to change directory to");
|
|
217 if (!(mp = m_gmsg (folder)))
|
|
218 adios (NULLCP, "unable to read folder %s", folder);
|
|
219 if (mp -> hghmsg == 0)
|
|
220 adios (NULLCP, "no messages in %s", folder);
|
|
221
|
|
222 for (msgnum = 0; msgnum < msgp; msgnum++)
|
|
223 if (!m_convert (mp, msgs[msgnum]))
|
|
224 done (1);
|
|
225 m_setseq (mp);
|
|
226
|
|
227 if (seqp == 0)
|
|
228 listsw++;
|
|
229 if (publicsw == -1)
|
|
230 publicsw = mp -> msgflags & READONLY ? 0 : 1;
|
|
231 if (publicsw && (mp -> msgflags & READONLY))
|
|
232 adios (NULLCP, "folder %s is read-only, so -public not allowed",
|
|
233 folder);
|
|
234
|
|
235 /* */
|
|
236
|
|
237 if (!pcompile (vec, NULLCP))
|
|
238 done (1);
|
|
239
|
|
240 lo = mp -> lowsel;
|
|
241 hi = mp -> hghsel;
|
|
242
|
|
243 for (msgnum = mp -> lowsel; msgnum <= mp -> hghsel; msgnum++)
|
|
244 if (mp -> msgstats[msgnum] & SELECTED) {
|
|
245 if ((fp = fopen (cp = m_name (msgnum), "r")) == NULL)
|
|
246 admonish (cp, "unable to read message");
|
|
247 if (fp && pmatches (fp, msgnum, 0L, 0L)) {
|
|
248 if (msgnum < lo)
|
|
249 lo = msgnum;
|
|
250 if (msgnum > hi)
|
|
251 hi = msgnum;
|
|
252 }
|
|
253 else {
|
|
254 mp -> msgstats[msgnum] &= ~SELECTED;
|
|
255 mp -> numsel--;
|
|
256 }
|
|
257 if (fp)
|
|
258 (void) fclose (fp);
|
|
259 }
|
|
260
|
|
261 mp -> lowsel = lo;
|
|
262 mp -> hghsel = hi;
|
|
263
|
|
264 if (mp -> numsel <= 0)
|
|
265 adios (NULLCP, "no messages match specification");
|
|
266
|
|
267 /* */
|
|
268
|
|
269 seqs[seqp] = NULL;
|
|
270 for (seqp = 0; seqs[seqp]; seqp++) {
|
|
271 if (zerosw && !m_seqnew (mp, seqs[seqp], publicsw))
|
|
272 done (1);
|
|
273 for (msgnum = mp -> lowsel; msgnum <= mp -> hghsel; msgnum++)
|
|
274 if (mp -> msgstats[msgnum] & SELECTED)
|
|
275 if (!m_seqadd (mp, seqs[seqp], msgnum, publicsw))
|
|
276 done (1);
|
|
277 }
|
|
278
|
|
279 if (listsw) {
|
|
280 for (msgnum = mp -> lowsel; msgnum <= mp -> hghsel; msgnum++)
|
|
281 if (mp -> msgstats[msgnum] & SELECTED)
|
|
282 printf ("%s\n", m_name (msgnum));
|
|
283 }
|
|
284 else
|
|
285 printf ("%d hit%s\n", mp -> numsel,
|
|
286 mp -> numsel == 1 ? "" : "s");
|
|
287
|
|
288 m_replace (pfolder, folder);
|
|
289 m_sync (mp);
|
|
290 m_update ();
|
|
291
|
|
292 done (0);
|
|
293 }
|
|
294
|
|
295 /* */
|
|
296
|
|
297 void done (status)
|
|
298 int status;
|
|
299 {
|
|
300 if (listsw && status && !isatty (fileno (stdout)))
|
|
301 printf ("0\n");
|
|
302 exit (status);
|
|
303 }
|