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