Mercurial > emacs
annotate lib-src/movemail.c @ 5381:56db0930185b
Added `kibo', who actually greps for his name on Usenet.
author | Noah Friedman <friedman@splode.com> |
---|---|
date | Fri, 31 Dec 1993 05:12:51 +0000 |
parents | 60fa1ee0c98c |
children | d7408b99b88f |
rev | line source |
---|---|
23 | 1 /* movemail foo bar -- move file foo to file bar, |
2 locking file foo the way /bin/mail respects. | |
580 | 3 Copyright (C) 1986, 1992 Free Software Foundation, Inc. |
23 | 4 |
5 This file is part of GNU Emacs. | |
6 | |
38 | 7 GNU Emacs is free software; you can redistribute it and/or modify |
8 it under the terms of the GNU General Public License as published by | |
9 the Free Software Foundation; either version 1, or (at your option) | |
10 any later version. | |
23 | 11 |
38 | 12 GNU Emacs is distributed in the hope that it will be useful, |
13 but WITHOUT ANY WARRANTY; without even the implied warranty of | |
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
15 GNU General Public License for more details. | |
16 | |
17 You should have received a copy of the GNU General Public License | |
18 along with GNU Emacs; see the file COPYING. If not, write to | |
19 the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ | |
23 | 20 |
510
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
21 /* Important notice: defining MAIL_USE_FLOCK *will cause loss of mail* |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
22 if you do it on a system that does not normally use flock as its way of |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
23 interlocking access to inbox files. The setting of MAIL_USE_FLOCK |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
24 *must agree* with the system's own conventions. |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
25 It is not a choice that is up to you. |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
26 |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
27 So, if your system uses lock files rather than flock, then the only way |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
28 you can get proper operation is to enable movemail to write lockfiles there. |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
29 This means you must either give that directory access modes |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
30 that permit everyone to write lockfiles in it, or you must make movemail |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
31 a setuid or setgid program. */ |
4eaef1578a15
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
447
diff
changeset
|
32 |
23 | 33 /* |
34 * Modified January, 1986 by Michael R. Gretzinger (Project Athena) | |
35 * | |
36 * Added POP (Post Office Protocol) service. When compiled -DPOP | |
37 * movemail will accept input filename arguments of the form | |
38 * "po:username". This will cause movemail to open a connection to | |
39 * a pop server running on $MAILHOST (environment variable). Movemail | |
40 * must be setuid to root in order to work with POP. | |
41 * | |
42 * New module: popmail.c | |
43 * Modified routines: | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
44 * main - added code within #ifdef MAIL_USE_POP; added setuid (getuid ()) |
23 | 45 * after POP code. |
46 * New routines in movemail.c: | |
47 * get_errmsg - return pointer to system error message | |
48 * | |
49 */ | |
50 | |
51 #include <sys/types.h> | |
52 #include <sys/stat.h> | |
53 #include <sys/file.h> | |
54 #include <errno.h> | |
55 #define NO_SHORTNAMES /* Tell config not to load remap.h */ | |
4696
1fc792473491
Include <config.h> instead of "config.h".
Roland McGrath <roland@gnu.org>
parents:
3309
diff
changeset
|
56 #include <../src/config.h> |
23 | 57 |
58 #ifdef USG | |
59 #include <fcntl.h> | |
60 #include <unistd.h> | |
27 | 61 #ifndef F_OK |
62 #define F_OK 0 | |
63 #define X_OK 1 | |
64 #define W_OK 2 | |
65 #define R_OK 4 | |
66 #endif | |
23 | 67 #endif /* USG */ |
68 | |
69 #ifdef XENIX | |
70 #include <sys/locking.h> | |
71 #endif | |
72 | |
25 | 73 #ifdef MAIL_USE_MMDF |
74 extern int lk_open (), lk_close (); | |
75 #endif | |
76 | |
23 | 77 /* Cancel substitutions made by config.h for Emacs. */ |
78 #undef open | |
79 #undef read | |
80 #undef write | |
81 #undef close | |
82 | |
571 | 83 char *malloc (); |
733 | 84 char *strcpy (); |
23 | 85 char *concat (); |
571 | 86 char *xmalloc (); |
87 #ifndef errno | |
23 | 88 extern int errno; |
571 | 89 #endif |
23 | 90 |
91 /* Nonzero means this is name of a lock file to delete on fatal error. */ | |
92 char *delete_lockname; | |
93 | |
94 main (argc, argv) | |
95 int argc; | |
96 char **argv; | |
97 { | |
98 char *inname, *outname; | |
99 int indesc, outdesc; | |
100 int nread; | |
101 | |
102 #ifndef MAIL_USE_FLOCK | |
103 struct stat st; | |
104 long now; | |
105 int tem; | |
106 char *lockname, *p; | |
601
3db1540d4b97
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
580
diff
changeset
|
107 char *tempname; |
23 | 108 int desc; |
109 #endif /* not MAIL_USE_FLOCK */ | |
110 | |
111 delete_lockname = 0; | |
112 | |
113 if (argc < 3) | |
114 fatal ("two arguments required"); | |
115 | |
116 inname = argv[1]; | |
117 outname = argv[2]; | |
118 | |
25 | 119 #ifdef MAIL_USE_MMDF |
120 mmdf_init (argv[0]); | |
121 #endif | |
122 | |
120 | 123 /* Check access to output file. */ |
23 | 124 if (access (outname, F_OK) == 0 && access (outname, W_OK) != 0) |
125 pfatal_with_name (outname); | |
126 | |
127 /* Also check that outname's directory is writeable to the real uid. */ | |
128 { | |
129 char *buf = (char *) malloc (strlen (outname) + 1); | |
130 char *p, q; | |
131 strcpy (buf, outname); | |
132 p = buf + strlen (buf); | |
133 while (p > buf && p[-1] != '/') | |
134 *--p = 0; | |
135 if (p == buf) | |
136 *p++ = '.'; | |
137 if (access (buf, W_OK) != 0) | |
138 pfatal_with_name (buf); | |
139 free (buf); | |
140 } | |
141 | |
142 #ifdef MAIL_USE_POP | |
3309
f00054d40753
* movemail.c [MAIL_USE_POP] (main): Don't use non-portable
Jim Blandy <jimb@redhat.com>
parents:
733
diff
changeset
|
143 if (!strncmp (inname, "po:", 3)) |
23 | 144 { |
145 int status; char *user; | |
146 | |
3309
f00054d40753
* movemail.c [MAIL_USE_POP] (main): Don't use non-portable
Jim Blandy <jimb@redhat.com>
parents:
733
diff
changeset
|
147 for (user = &inname[strlen (inname) - 1]; user >= inname; user--) |
f00054d40753
* movemail.c [MAIL_USE_POP] (main): Don't use non-portable
Jim Blandy <jimb@redhat.com>
parents:
733
diff
changeset
|
148 if (*user == ':') |
f00054d40753
* movemail.c [MAIL_USE_POP] (main): Don't use non-portable
Jim Blandy <jimb@redhat.com>
parents:
733
diff
changeset
|
149 break; |
f00054d40753
* movemail.c [MAIL_USE_POP] (main): Don't use non-portable
Jim Blandy <jimb@redhat.com>
parents:
733
diff
changeset
|
150 |
23 | 151 status = popmail (user, outname); |
152 exit (status); | |
153 } | |
154 | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
155 setuid (getuid ()); |
23 | 156 #endif /* MAIL_USE_POP */ |
157 | |
120 | 158 /* Check access to input file. */ |
159 if (access (inname, R_OK | W_OK) != 0) | |
160 pfatal_with_name (inname); | |
161 | |
25 | 162 #ifndef MAIL_USE_MMDF |
23 | 163 #ifndef MAIL_USE_FLOCK |
164 /* Use a lock file named /usr/spool/mail/$USER.lock: | |
165 If it exists, the mail file is locked. */ | |
351
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
166 /* Note: this locking mechanism is *required* by the mailer |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
167 (on systems which use it) to prevent loss of mail. |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
168 |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
169 On systems that use a lock file, extracting the mail without locking |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
170 WILL occasionally cause loss of mail due to timing errors! |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
171 |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
172 So, if creation of the lock file fails |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
173 due to access permission on /usr/spool/mail, |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
174 you simply MUST change the permission |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
175 and/or make movemail a setgid program |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
176 so it can create lock files properly. |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
177 |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
178 You might also wish to verify that your system is one |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
179 which uses lock files for this purpose. Some systems use other methods. |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
180 |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
181 If your system uses the `flock' system call for mail locking, |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
182 define MAIL_USE_FLOCK in config.h or the s-*.h file |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
183 and recompile movemail. If the s- file for your system |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
184 should define MAIL_USE_FLOCK but does not, send a bug report |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
185 to bug-gnu-emacs@prep.ai.mit.edu so we can fix it. */ |
5729b1cc3942
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
127
diff
changeset
|
186 |
23 | 187 lockname = concat (inname, ".lock", ""); |
5318
60fa1ee0c98c
(main): When making tempname, cast result of xmalloc.
Richard M. Stallman <rms@gnu.org>
parents:
4985
diff
changeset
|
188 tempname = (char *) xmalloc (strlen (inname) + strlen ("EXXXXXX") + 1); |
60fa1ee0c98c
(main): When making tempname, cast result of xmalloc.
Richard M. Stallman <rms@gnu.org>
parents:
4985
diff
changeset
|
189 strcpy (tempname, inname); |
23 | 190 p = tempname + strlen (tempname); |
191 while (p != tempname && p[-1] != '/') | |
192 p--; | |
193 *p = 0; | |
194 strcpy (p, "EXXXXXX"); | |
195 mktemp (tempname); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
196 unlink (tempname); |
23 | 197 |
198 while (1) | |
199 { | |
200 /* Create the lock file, but not under the lock file name. */ | |
201 /* Give up if cannot do that. */ | |
202 desc = open (tempname, O_WRONLY | O_CREAT, 0666); | |
203 if (desc < 0) | |
4985
623be452ebb8
(main): Fix error message text.
Richard M. Stallman <rms@gnu.org>
parents:
4696
diff
changeset
|
204 pfatal_with_name ("lock file--see source file lib-src/movemail.c"); |
23 | 205 close (desc); |
206 | |
207 tem = link (tempname, lockname); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
208 unlink (tempname); |
23 | 209 if (tem >= 0) |
210 break; | |
211 sleep (1); | |
212 | |
213 /* If lock file is a minute old, unlock it. */ | |
214 if (stat (lockname, &st) >= 0) | |
215 { | |
216 now = time (0); | |
217 if (st.st_ctime < now - 60) | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
218 unlink (lockname); |
23 | 219 } |
220 } | |
221 | |
222 delete_lockname = lockname; | |
223 #endif /* not MAIL_USE_FLOCK */ | |
224 | |
225 #ifdef MAIL_USE_FLOCK | |
226 indesc = open (inname, O_RDWR); | |
227 #else /* if not MAIL_USE_FLOCK */ | |
228 indesc = open (inname, O_RDONLY); | |
229 #endif /* not MAIL_USE_FLOCK */ | |
25 | 230 #else /* MAIL_USE_MMDF */ |
231 indesc = lk_open (inname, O_RDONLY, 0, 0, 10); | |
232 #endif /* MAIL_USE_MMDF */ | |
233 | |
23 | 234 if (indesc < 0) |
235 pfatal_with_name (inname); | |
236 | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
237 #if defined (BSD) || defined (XENIX) |
23 | 238 /* In case movemail is setuid to root, make sure the user can |
239 read the output file. */ | |
240 /* This is desirable for all systems | |
241 but I don't want to assume all have the umask system call */ | |
242 umask (umask (0) & 0333); | |
243 #endif /* BSD or Xenix */ | |
244 outdesc = open (outname, O_WRONLY | O_CREAT | O_EXCL, 0666); | |
245 if (outdesc < 0) | |
246 pfatal_with_name (outname); | |
247 #ifdef MAIL_USE_FLOCK | |
248 #ifdef XENIX | |
249 if (locking (indesc, LK_RLCK, 0L) < 0) pfatal_with_name (inname); | |
250 #else | |
251 if (flock (indesc, LOCK_EX) < 0) pfatal_with_name (inname); | |
252 #endif | |
253 #endif /* MAIL_USE_FLOCK */ | |
254 | |
604 | 255 { |
256 char buf[1024]; | |
257 | |
258 while (1) | |
259 { | |
260 nread = read (indesc, buf, sizeof buf); | |
261 if (nread != write (outdesc, buf, nread)) | |
262 { | |
263 int saved_errno = errno; | |
264 unlink (outname); | |
265 errno = saved_errno; | |
266 pfatal_with_name (outname); | |
267 } | |
268 if (nread < sizeof buf) | |
269 break; | |
270 } | |
271 } | |
23 | 272 |
273 #ifdef BSD | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
274 if (fsync (outdesc) < 0) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
275 pfatal_and_delete (outname); |
23 | 276 #endif |
277 | |
278 /* Check to make sure no errors before we zap the inbox. */ | |
279 if (close (outdesc) != 0) | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
280 pfatal_and_delete (outname); |
23 | 281 |
282 #ifdef MAIL_USE_FLOCK | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
283 #if defined (STRIDE) || defined (XENIX) |
23 | 284 /* Stride, xenix have file locking, but no ftruncate. This mess will do. */ |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
285 close (open (inname, O_CREAT | O_TRUNC | O_RDWR, 0666)); |
23 | 286 #else |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
287 ftruncate (indesc, 0L); |
23 | 288 #endif /* STRIDE or XENIX */ |
289 #endif /* MAIL_USE_FLOCK */ | |
25 | 290 |
291 #ifdef MAIL_USE_MMDF | |
292 lk_close (indesc, 0, 0, 0); | |
293 #else | |
23 | 294 close (indesc); |
25 | 295 #endif |
23 | 296 |
297 #ifndef MAIL_USE_FLOCK | |
298 /* Delete the input file; if we can't, at least get rid of its contents. */ | |
571 | 299 #ifdef MAIL_UNLINK_SPOOL |
300 /* This is generally bad to do, because it destroys the permissions | |
301 that were set on the file. Better to just empty the file. */ | |
302 if (unlink (inname) < 0 && errno != ENOENT) | |
303 #endif /* MAIL_UNLINK_SPOOL */ | |
304 creat (inname, 0600); | |
25 | 305 #ifndef MAIL_USE_MMDF |
306 unlink (lockname); | |
307 #endif /* not MAIL_USE_MMDF */ | |
23 | 308 #endif /* not MAIL_USE_FLOCK */ |
309 exit (0); | |
310 } | |
311 | |
312 /* Print error message and exit. */ | |
313 | |
314 fatal (s1, s2) | |
315 char *s1, *s2; | |
316 { | |
317 if (delete_lockname) | |
318 unlink (delete_lockname); | |
319 error (s1, s2); | |
320 exit (1); | |
321 } | |
322 | |
323 /* Print error message. `s1' is printf control string, `s2' is arg for it. */ | |
324 | |
120 | 325 error (s1, s2, s3) |
326 char *s1, *s2, *s3; | |
23 | 327 { |
328 printf ("movemail: "); | |
120 | 329 printf (s1, s2, s3); |
23 | 330 printf ("\n"); |
331 } | |
332 | |
333 pfatal_with_name (name) | |
334 char *name; | |
335 { | |
336 extern int errno, sys_nerr; | |
337 extern char *sys_errlist[]; | |
338 char *s; | |
339 | |
340 if (errno < sys_nerr) | |
341 s = concat ("", sys_errlist[errno], " for %s"); | |
342 else | |
343 s = "cannot open %s"; | |
344 fatal (s, name); | |
345 } | |
346 | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
347 pfatal_and_delete (name) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
348 char *name; |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
349 { |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
350 extern int errno, sys_nerr; |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
351 extern char *sys_errlist[]; |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
352 char *s; |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
353 |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
354 if (errno < sys_nerr) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
355 s = concat ("", sys_errlist[errno], " for %s"); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
356 else |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
357 s = "cannot open %s"; |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
358 |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
359 unlink (name); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
360 fatal (s, name); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
361 } |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
362 |
23 | 363 /* Return a newly-allocated string whose contents concatenate those of s1, s2, s3. */ |
364 | |
365 char * | |
366 concat (s1, s2, s3) | |
367 char *s1, *s2, *s3; | |
368 { | |
369 int len1 = strlen (s1), len2 = strlen (s2), len3 = strlen (s3); | |
370 char *result = (char *) xmalloc (len1 + len2 + len3 + 1); | |
371 | |
372 strcpy (result, s1); | |
373 strcpy (result + len1, s2); | |
374 strcpy (result + len1 + len2, s3); | |
375 *(result + len1 + len2 + len3) = 0; | |
376 | |
377 return result; | |
378 } | |
379 | |
380 /* Like malloc but get fatal error if memory is exhausted. */ | |
381 | |
571 | 382 char * |
23 | 383 xmalloc (size) |
571 | 384 unsigned size; |
23 | 385 { |
571 | 386 char *result = malloc (size); |
23 | 387 if (!result) |
388 fatal ("virtual memory exhausted", 0); | |
389 return result; | |
390 } | |
391 | |
392 /* This is the guts of the interface to the Post Office Protocol. */ | |
393 | |
394 #ifdef MAIL_USE_POP | |
395 | |
396 #include <sys/socket.h> | |
397 #include <netinet/in.h> | |
398 #include <netdb.h> | |
399 #include <stdio.h> | |
634
52d0ff659265
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
604
diff
changeset
|
400 #include <pwd.h> |
23 | 401 |
402 #ifdef USG | |
403 #include <fcntl.h> | |
404 /* Cancel substitutions made by config.h for Emacs. */ | |
405 #undef open | |
406 #undef read | |
407 #undef write | |
408 #undef close | |
409 #endif /* USG */ | |
410 | |
411 #define NOTOK (-1) | |
412 #define OK 0 | |
413 #define DONE 1 | |
414 | |
415 char *progname; | |
416 FILE *sfi; | |
417 FILE *sfo; | |
418 char Errmsg[80]; | |
419 | |
420 static int debug = 0; | |
421 | |
120 | 422 char *get_errmsg (); |
423 char *getenv (); | |
424 int mbx_write (); | |
425 | |
426 popmail (user, outfile) | |
427 char *user; | |
428 char *outfile; | |
23 | 429 { |
120 | 430 char *host; |
431 int nmsgs, nbytes; | |
432 char response[128]; | |
433 register int i; | |
434 int mbfi; | |
435 FILE *mbf; | |
634
52d0ff659265
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
604
diff
changeset
|
436 struct passwd *pw = (struct passwd *) getpwuid (getuid ()); |
52d0ff659265
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
604
diff
changeset
|
437 if (pw == NULL) |
52d0ff659265
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
604
diff
changeset
|
438 fatal ("cannot determine user name"); |
23 | 439 |
120 | 440 host = getenv ("MAILHOST"); |
441 if (host == NULL) | |
442 { | |
443 fatal ("no MAILHOST defined"); | |
23 | 444 } |
445 | |
120 | 446 if (pop_init (host) == NOTOK) |
447 { | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
448 fatal (Errmsg); |
23 | 449 } |
450 | |
120 | 451 if (getline (response, sizeof response, sfi) != OK) |
452 { | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
453 fatal (response); |
23 | 454 } |
455 | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
456 if (pop_command ("USER %s", user) == NOTOK |
634
52d0ff659265
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
604
diff
changeset
|
457 || pop_command ("RPOP %s", pw->pw_name) == NOTOK) |
120 | 458 { |
459 pop_command ("QUIT"); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
460 fatal (Errmsg); |
23 | 461 } |
462 | |
120 | 463 if (pop_stat (&nmsgs, &nbytes) == NOTOK) |
464 { | |
465 pop_command ("QUIT"); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
466 fatal (Errmsg); |
120 | 467 } |
23 | 468 |
120 | 469 if (!nmsgs) |
470 { | |
471 pop_command ("QUIT"); | |
472 return 0; | |
23 | 473 } |
474 | |
120 | 475 mbfi = open (outfile, O_WRONLY | O_CREAT | O_EXCL, 0666); |
476 if (mbfi < 0) | |
477 { | |
478 pop_command ("QUIT"); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
479 pfatal_and_delete (outfile); |
120 | 480 } |
481 fchown (mbfi, getuid (), -1); | |
482 | |
483 if ((mbf = fdopen (mbfi, "w")) == NULL) | |
484 { | |
485 pop_command ("QUIT"); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
486 pfatal_and_delete (outfile); |
120 | 487 } |
488 | |
489 for (i = 1; i <= nmsgs; i++) | |
490 { | |
491 mbx_delimit_begin (mbf); | |
492 if (pop_retr (i, mbx_write, mbf) != OK) | |
493 { | |
494 pop_command ("QUIT"); | |
495 close (mbfi); | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
496 unlink (outfile); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
497 fatal (Errmsg); |
120 | 498 } |
499 mbx_delimit_end (mbf); | |
500 fflush (mbf); | |
501 } | |
502 | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
503 if (fsync (mbfi) < 0) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
504 { |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
505 pop_command ("QUIT"); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
506 pfatal_and_delete (outfile); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
507 } |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
508 |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
509 if (close (mbfi) == -1) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
510 { |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
511 pop_command ("QUIT"); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
512 pfatal_and_delete (outfile); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
513 } |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
514 |
120 | 515 for (i = 1; i <= nmsgs; i++) |
516 { | |
517 if (pop_command ("DELE %d", i) == NOTOK) | |
518 { | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
519 /* Better to ignore this failure. */ |
23 | 520 } |
521 } | |
522 | |
120 | 523 pop_command ("QUIT"); |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
524 return (0); |
23 | 525 } |
526 | |
120 | 527 pop_init (host) |
528 char *host; | |
23 | 529 { |
120 | 530 register struct hostent *hp; |
531 register struct servent *sp; | |
532 int lport = IPPORT_RESERVED - 1; | |
533 struct sockaddr_in sin; | |
534 register int s; | |
23 | 535 |
120 | 536 hp = gethostbyname (host); |
537 if (hp == NULL) | |
538 { | |
539 sprintf (Errmsg, "MAILHOST unknown: %s", host); | |
540 return NOTOK; | |
23 | 541 } |
542 | |
120 | 543 sp = getservbyname ("pop", "tcp"); |
544 if (sp == 0) | |
545 { | |
546 strcpy (Errmsg, "tcp/pop: unknown service"); | |
547 return NOTOK; | |
23 | 548 } |
549 | |
120 | 550 sin.sin_family = hp->h_addrtype; |
551 bcopy (hp->h_addr, (char *)&sin.sin_addr, hp->h_length); | |
552 sin.sin_port = sp->s_port; | |
553 s = rresvport (&lport); | |
554 if (s < 0) | |
555 { | |
556 sprintf (Errmsg, "error creating socket: %s", get_errmsg ()); | |
557 return NOTOK; | |
23 | 558 } |
559 | |
120 | 560 if (connect (s, (char *)&sin, sizeof sin) < 0) |
561 { | |
562 sprintf (Errmsg, "error during connect: %s", get_errmsg ()); | |
563 close (s); | |
564 return NOTOK; | |
23 | 565 } |
566 | |
120 | 567 sfi = fdopen (s, "r"); |
568 sfo = fdopen (s, "w"); | |
569 if (sfi == NULL || sfo == NULL) | |
570 { | |
571 sprintf (Errmsg, "error in fdopen: %s", get_errmsg ()); | |
572 close (s); | |
573 return NOTOK; | |
23 | 574 } |
575 | |
120 | 576 return OK; |
23 | 577 } |
578 | |
120 | 579 pop_command (fmt, a, b, c, d) |
580 char *fmt; | |
23 | 581 { |
120 | 582 char buf[128]; |
583 char errmsg[64]; | |
584 | |
585 sprintf (buf, fmt, a, b, c, d); | |
23 | 586 |
120 | 587 if (debug) fprintf (stderr, "---> %s\n", buf); |
588 if (putline (buf, Errmsg, sfo) == NOTOK) return NOTOK; | |
23 | 589 |
120 | 590 if (getline (buf, sizeof buf, sfi) != OK) |
591 { | |
592 strcpy (Errmsg, buf); | |
593 return NOTOK; | |
23 | 594 } |
595 | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
596 if (debug) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
597 fprintf (stderr, "<--- %s\n", buf); |
120 | 598 if (*buf != '+') |
599 { | |
600 strcpy (Errmsg, buf); | |
601 return NOTOK; | |
602 } | |
603 else | |
604 { | |
605 return OK; | |
23 | 606 } |
607 } | |
608 | |
609 | |
120 | 610 pop_stat (nmsgs, nbytes) |
611 int *nmsgs, *nbytes; | |
23 | 612 { |
120 | 613 char buf[128]; |
23 | 614 |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
615 if (debug) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
616 fprintf (stderr, "---> STAT\n"); |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
617 if (putline ("STAT", Errmsg, sfo) == NOTOK) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
618 return NOTOK; |
23 | 619 |
120 | 620 if (getline (buf, sizeof buf, sfi) != OK) |
621 { | |
622 strcpy (Errmsg, buf); | |
623 return NOTOK; | |
23 | 624 } |
625 | |
120 | 626 if (debug) fprintf (stderr, "<--- %s\n", buf); |
627 if (*buf != '+') | |
628 { | |
629 strcpy (Errmsg, buf); | |
630 return NOTOK; | |
631 } | |
632 else | |
633 { | |
634 sscanf (buf, "+OK %d %d", nmsgs, nbytes); | |
635 return OK; | |
23 | 636 } |
637 } | |
638 | |
120 | 639 pop_retr (msgno, action, arg) |
640 int (*action)(); | |
23 | 641 { |
120 | 642 char buf[128]; |
23 | 643 |
120 | 644 sprintf (buf, "RETR %d", msgno); |
645 if (debug) fprintf (stderr, "%s\n", buf); | |
646 if (putline (buf, Errmsg, sfo) == NOTOK) return NOTOK; | |
23 | 647 |
120 | 648 if (getline (buf, sizeof buf, sfi) != OK) |
649 { | |
650 strcpy (Errmsg, buf); | |
651 return NOTOK; | |
23 | 652 } |
653 | |
120 | 654 while (1) |
655 { | |
656 switch (multiline (buf, sizeof buf, sfi)) | |
657 { | |
23 | 658 case OK: |
120 | 659 (*action)(buf, arg); |
660 break; | |
23 | 661 case DONE: |
120 | 662 return OK; |
23 | 663 case NOTOK: |
120 | 664 strcpy (Errmsg, buf); |
665 return NOTOK; | |
23 | 666 } |
667 } | |
668 } | |
669 | |
120 | 670 getline (buf, n, f) |
671 char *buf; | |
672 register int n; | |
673 FILE *f; | |
23 | 674 { |
120 | 675 register char *p; |
676 int c; | |
23 | 677 |
120 | 678 p = buf; |
679 while (--n > 0 && (c = fgetc (f)) != EOF) | |
680 if ((*p++ = c) == '\n') break; | |
23 | 681 |
120 | 682 if (ferror (f)) |
683 { | |
684 strcpy (buf, "error on connection"); | |
685 return NOTOK; | |
686 } | |
687 | |
688 if (c == EOF && p == buf) | |
689 { | |
690 strcpy (buf, "connection closed by foreign host"); | |
691 return DONE; | |
23 | 692 } |
693 | |
120 | 694 *p = NULL; |
695 if (*--p == '\n') *p = NULL; | |
696 if (*--p == '\r') *p = NULL; | |
697 return OK; | |
23 | 698 } |
699 | |
120 | 700 multiline (buf, n, f) |
701 char *buf; | |
702 register int n; | |
703 FILE *f; | |
23 | 704 { |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
705 if (getline (buf, n, f) != OK) |
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
706 return NOTOK; |
120 | 707 if (*buf == '.') |
708 { | |
709 if (*(buf+1) == NULL) | |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
710 return DONE; |
120 | 711 else |
447
2e226dcdaf0f
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
351
diff
changeset
|
712 strcpy (buf, buf+1); |
23 | 713 } |
120 | 714 return OK; |
23 | 715 } |
716 | |
717 char * | |
120 | 718 get_errmsg () |
23 | 719 { |
120 | 720 extern int errno, sys_nerr; |
721 extern char *sys_errlist[]; | |
722 char *s; | |
23 | 723 |
120 | 724 if (errno < sys_nerr) |
725 s = sys_errlist[errno]; | |
726 else | |
727 s = "unknown error"; | |
728 return (s); | |
23 | 729 } |
730 | |
120 | 731 putline (buf, err, f) |
732 char *buf; | |
733 char *err; | |
734 FILE *f; | |
23 | 735 { |
120 | 736 fprintf (f, "%s\r\n", buf); |
737 fflush (f); | |
738 if (ferror (f)) | |
739 { | |
740 strcpy (err, "lost connection"); | |
741 return NOTOK; | |
23 | 742 } |
120 | 743 return OK; |
23 | 744 } |
745 | |
120 | 746 mbx_write (line, mbf) |
747 char *line; | |
748 FILE *mbf; | |
23 | 749 { |
120 | 750 fputs (line, mbf); |
751 fputc (0x0a, mbf); | |
23 | 752 } |
753 | |
120 | 754 mbx_delimit_begin (mbf) |
755 FILE *mbf; | |
23 | 756 { |
127
762710f7381a
*** empty log message ***
Richard M. Stallman <rms@gnu.org>
parents:
120
diff
changeset
|
757 fputs ("\f\n0, unseen,,\n", mbf); |
23 | 758 } |
759 | |
120 | 760 mbx_delimit_end (mbf) |
761 FILE *mbf; | |
23 | 762 { |
120 | 763 putc ('\037', mbf); |
23 | 764 } |
765 | |
766 #endif /* MAIL_USE_POP */ |