2 * S-nail - a mail user agent derived from Berkeley Mail.
4 * Copyright (c) 2000-2004 Gunnar Ritter, Freiburg i. Br., Germany.
5 * Copyright (c) 2012 Steffen "Daode" Nurpmeso.
8 * Copyright (c) 1980, 1993
9 * The Regents of the University of California. All rights reserved.
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
14 * 1. Redistributions of source code must retain the above copyright
15 * notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 * notice, this list of conditions and the following disclaimer in the
18 * documentation and/or other materials provided with the distribution.
19 * 3. All advertising materials mentioning features or use of this software
20 * must display the following acknowledgement:
21 * This product includes software developed by the University of
22 * California, Berkeley and its contributors.
23 * 4. Neither the name of the University nor the names of its contributors
24 * may be used to endorse or promote products derived from this software
25 * without specific prior written permission.
27 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
28 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
31 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
32 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
33 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
34 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
35 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
36 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
48 * Mail -- a mail program
50 * Still more user commands.
53 static int bangexp(char **str
, size_t *size
);
54 static void make_ref_and_cs(struct message
*mp
, struct header
*head
);
55 static int (*respond_or_Respond(int c
))(int *, int);
56 static int respond_internal(int *msgvec
, int recipient_record
);
57 static char *reedit(char *subj
);
58 static char *fwdedit(char *subj
);
59 static void onpipe(int signo
);
60 static void asort(char **list
);
61 static int diction(const void *a
, const void *b
);
62 static int file1(char *name
);
63 static int shellecho(const char *cp
);
64 static int Respond_internal(int *msgvec
, int recipient_record
);
65 static int resend1(void *v
, int add_resent
);
66 static void list_shortcuts(void);
67 static enum okay
delete_shortcut(const char *str
);
68 static float huge(void);
71 * Process a shell escape by saving signals, ignoring signals,
78 sighandler_type sigint
= safe_signal(SIGINT
, SIG_IGN
);
83 cmd
= smalloc(cmdsize
= strlen(str
) + 1);
85 if (bangexp(&cmd
, &cmdsize
) < 0)
87 if ((shell
= value("SHELL")) == NULL
)
89 run_command(shell
, 0, -1, -1, "-c", cmd
, NULL
);
90 safe_signal(SIGINT
, sigint
);
97 * Fork an interactive shell.
103 sighandler_type sigint
= safe_signal(SIGINT
, SIG_IGN
);
107 if ((shell
= value("SHELL")) == NULL
)
109 run_command(shell
, 0, -1, -1, NULL
, NULL
, NULL
);
110 safe_signal(SIGINT
, sigint
);
116 * Expand the shell escape by expanding unescaped !'s into the
117 * last issued command where possible.
120 static char *lastbang
;
121 static size_t lastbangsize
;
124 bangexp(char **str
, size_t *size
)
128 int dobang
= value("bang") != NULL
;
129 size_t sz
, i
, j
, bangbufsize
;
131 bangbuf
= smalloc(bangbufsize
= *size
);
135 if ((*str
)[i
] == '!') {
136 sz
= strlen(lastbang
);
137 bangbuf
= srealloc(bangbuf
, bangbufsize
+= sz
);
139 strcpy(&bangbuf
[j
], lastbang
);
145 if ((*str
)[i
] == '\\' && (*str
)[i
+ 1] == '!') {
150 bangbuf
[j
++] = (*str
)[i
++];
154 printf("!%s\n", bangbuf
);
159 *str
= srealloc(*str
, *size
= sz
+ 1);
160 strcpy(*str
, bangbuf
);
161 if (sz
>= lastbangsize
)
162 lastbang
= srealloc(lastbang
, lastbangsize
= sz
+ 1);
163 strcpy(lastbang
, bangbuf
);
173 /* Very ugly, but take care for compiler supported string lengths :( */
178 "type <message list> type messages\n"
179 "next goto and type next message\n"
180 "from <message list> give head lines of messages\n"
181 "headers print out active message headers\n"
182 "delete <message list> delete messages\n"
183 "undelete <message list> undelete messages\n");
185 "save <message list> folder append messages to folder and mark as saved\n"
186 "copy <message list> folder append messages to folder without marking them\n"
187 "write <message list> file append message texts to file, save attachments\n"
188 "preserve <message list> keep incoming messages in mailbox even if saved\n"
189 "Reply <message list> reply to message senders\n"
190 "reply <message list> reply to message senders and all recipients\n");
192 "mail addresses mail to specific recipients\n"
193 "file folder change to another folder\n"
194 "quit quit and apply changes to folder\n"
195 "xit quit and discard changes made to folder\n"
197 "cd <directory> chdir to directory or home if none given\n"
198 "list list names of all available commands\n");
200 "\nA <message list> consists of integers, ranges of same, or other criteria\n"
201 "separated by spaces. If omitted, %s uses the last message typed.\n",
207 * Change user's working directory.
215 if (*arglist
== NULL
)
218 if ((cp
= file_expand(*arglist
)) == NULL
)
228 make_ref_and_cs(struct message
*mp
, struct header
*head
)
230 char *oldref
, *oldmsgid
, *newref
, *cp
;
235 oldref
= hfield1("references", mp
);
236 oldmsgid
= hfield1("message-id", mp
);
237 if (oldmsgid
== NULL
|| *oldmsgid
== '\0') {
243 reflen
+= strlen(oldref
) + 2;
245 reflen
+= strlen(oldmsgid
);
246 newref
= ac_alloc(reflen
);
248 strcpy(newref
, oldref
);
250 strcat(newref
, ", ");
251 strcat(newref
, oldmsgid
);
254 strcpy(newref
, oldmsgid
);
255 n
= extract(newref
, GREF
);
258 * Limit the references to 21 entries.
260 while (n
->n_flink
!= NULL
)
262 for (i
= 1; i
< 21; i
++) {
263 if (n
->n_blink
!= NULL
)
270 if (value("reply-in-same-charset") != NULL
&&
271 (cp
= hfield1("content-type", mp
)) != NULL
)
272 head
->h_charset
= mime_getparam("charset", cp
);
276 (*respond_or_Respond(int c
))(int *, int)
280 opt
+= (value("Replyall") != NULL
);
281 opt
+= (value("flipr") != NULL
);
282 return ((opt
== 1) ^ (c
== 'R')) ? Respond_internal
: respond_internal
;
288 return (respond_or_Respond('r'))((int *)v
, 0);
294 return respond_internal((int *)v
, 0);
298 respondsender(void *v
)
300 return Respond_internal((int *)v
, 0);
306 return (respond_or_Respond('r'))((int *)v
, 1);
312 return respond_internal((int *)v
, 1);
316 followupsender(void *v
)
318 return Respond_internal((int *)v
, 1);
322 * Reply to a list of messages. Extract each name from the
323 * message header and send them off to mail1()
326 respond_internal(int *msgvec
, int recipient_record
)
331 enum gfield gf
= value("fullnames") ? GFULL
: GSKIN
;
332 struct name
*np
= NULL
;
335 memset(&head
, 0, sizeof head
);
336 if (msgvec
[1] != 0) {
337 printf(catgets(catd
, CATSET
, 37,
338 "Sorry, can't reply to multiple messages at once\n"));
341 mp
= &message
[msgvec
[0] - 1];
344 if ((rcv
= hfield1("reply-to", mp
)) == NULL
)
345 if ((rcv
= hfield1("from", mp
)) == NULL
)
348 np
= lextract(rcv
, GTO
|gf
);
349 if (! value("recipients-in-cc") && (cp
= hfield1("to", mp
)) != NULL
)
350 np
= cat(np
, lextract(cp
, GTO
|gf
));
353 * Delete my name from the reply list,
354 * and with it, all my alternate names.
356 np
= delete_alternates(np
);
358 np
= lextract(rcv
, GTO
|gf
);
360 head
.h_subject
= hfield1("subject", mp
);
361 head
.h_subject
= reedit(head
.h_subject
);
364 if (value("recipients-in-cc") && (cp
= hfield1("to", mp
)) != NULL
)
365 np
= lextract(cp
, GCC
|gf
);
366 if ((cp
= hfield1("cc", mp
)) != NULL
)
367 np
= cat(np
, lextract(cp
, GCC
|gf
));
369 head
.h_cc
= elide(delete_alternates(np
));
370 make_ref_and_cs(mp
, &head
);
371 Eflag
= value("skipemptybody") != NULL
;
372 if (mail1(&head
, 1, mp
, NULL
, recipient_record
, 0, 0, Eflag
) == OKAY
&&
373 value("markanswered") && (mp
->m_flag
& MANSWERED
) == 0)
374 mp
->m_flag
|= MANSWER
|MANSWERED
;
379 * Modify the subject we are replying to to begin with Re: if
380 * it does not already.
388 if (subj
== NULL
|| *subj
== '\0')
392 mime_fromhdr(&in
, &out
, TD_ISPR
|TD_ICONV
);
393 if ((out
.s
[0] == 'r' || out
.s
[0] == 'R') &&
394 (out
.s
[1] == 'e' || out
.s
[1] == 'E') &&
397 newsubj
= salloc(out
.l
+ 5);
398 strcpy(newsubj
, "Re: ");
399 strcpy(newsubj
+ 4, out
.s
);
404 * Forward a message to a new recipient, in the sense of RFC 2822.
407 forward1(char *str
, int recipient_record
)
414 int forward_as_attachment
;
416 forward_as_attachment
= value("forward-as-attachment") != NULL
;
417 msgvec
= salloc((msgCount
+ 2) * sizeof *msgvec
);
418 if ((recipient
= laststring(str
, &f
, 0)) == NULL
) {
419 puts(catgets(catd
, CATSET
, 47, "No recipient specified."));
423 *msgvec
= first(0, MMNORM
);
427 printf("No messages to forward.\n");
432 if (f
&& getmsglist(str
, msgvec
, 0) < 0)
437 printf("No applicable messages.\n");
440 if (msgvec
[1] != 0) {
441 printf("Cannot forward multiple messages at once\n");
444 memset(&head
, 0, sizeof head
);
445 if ((head
.h_to
= lextract(recipient
,
446 GTO
| (value("fullnames") ? GFULL
: GSKIN
))) == NULL
)
448 mp
= &message
[*msgvec
- 1];
449 if (forward_as_attachment
) {
450 head
.h_attach
= csalloc(1, sizeof *head
.h_attach
);
451 head
.h_attach
->a_msgno
= *msgvec
;
456 head
.h_subject
= hfield1("subject", mp
);
457 head
.h_subject
= fwdedit(head
.h_subject
);
458 Eflag
= value("skipemptybody") != NULL
;
459 mail1(&head
, 1, forward_as_attachment
? NULL
: mp
,
460 NULL
, recipient_record
, 1, 0, Eflag
);
465 * Modify the subject we are replying to to begin with Fwd:.
473 if (subj
== NULL
|| *subj
== '\0')
477 mime_fromhdr(&in
, &out
, TD_ISPR
|TD_ICONV
);
478 newsubj
= salloc(strlen(out
.s
) + 6);
479 strcpy(newsubj
, "Fwd: ");
480 strcpy(&newsubj
[5], out
.s
);
486 * The 'forward' command.
491 return forward1(v
, 0);
495 * Similar to forward, saving the message in a file named after the
501 return forward1(v
, 1);
505 * Preserve the named messages, so that they will be sent
506 * back to the system mailbox.
516 printf(catgets(catd
, CATSET
, 39,
517 "Cannot \"preserve\" in edit mode\n"));
520 for (ip
= msgvec
; *ip
!= 0; ip
++) {
522 mp
= &message
[mesg
-1];
523 mp
->m_flag
|= MPRESERVE
;
527 * This is now Austin Group Request XCU #20.
535 * Mark all given messages as unread.
543 for (ip
= msgvec
; *ip
!= 0; ip
++) {
544 setdot(&message
[*ip
-1]);
545 dot
->m_flag
&= ~(MREAD
|MTOUCH
);
546 dot
->m_flag
|= MSTATUS
;
547 if (mb
.mb_type
== MB_IMAP
|| mb
.mb_type
== MB_CACHE
)
548 imap_unread(&message
[*ip
-1], *ip
);
550 * The "unread" command is not part of POSIX mailx.
558 * Mark all given messages as read.
566 for (ip
= msgvec
; *ip
; ip
++) {
567 setdot(&message
[*ip
-1]);
568 touch(&message
[*ip
-1]);
574 * Print the size of each message.
583 for (ip
= msgvec
; *ip
!= 0; ip
++) {
585 mp
= &message
[mesg
-1];
586 printf("%d: ", mesg
);
587 if (mp
->m_xlines
> 0)
588 printf("%ld", mp
->m_xlines
);
591 printf("/%lu\n", (unsigned long)mp
->m_xsize
);
597 * Quit quickly. If we are sourcing, just pop the input level
598 * by returning an error.
611 static sigjmp_buf pipejmp
;
618 siglongjmp(pipejmp
, 1);
622 * Set or display a variable value. Syntax is similar to that
633 FILE *volatile obuf
= stdout
;
634 int volatile bsdset
= (value("bsdcompat") != NULL
||
635 value("bsdset") != NULL
);
637 if (*arglist
== NULL
) {
638 for (h
= 0, s
= 1; h
< HSHSIZE
; h
++)
639 for (vp
= variables
[h
]; vp
!= NULL
; vp
= vp
->v_link
)
642 ap
= (char **)salloc(s
* sizeof *ap
);
643 for (h
= 0, p
= ap
; h
< HSHSIZE
; h
++)
644 for (vp
= variables
[h
]; vp
!= NULL
; vp
= vp
->v_link
)
648 if (is_a_tty
[0] && is_a_tty
[1] && (cp
= value("crt")) != NULL
) {
649 if (s
> (*cp
== '\0' ? screensize() : atoi(cp
)) + 3) {
651 if (sigsetjmp(pipejmp
, 1))
653 if ((obuf
= Popen(cp
, "w", NULL
, 1)) == NULL
) {
657 safe_signal(SIGPIPE
, onpipe
);
660 for (p
= ap
; *p
!= NULL
; p
++) {
662 fprintf(obuf
, "%s\t%s\n", *p
, value(*p
));
664 if ((cp
= value(*p
)) != NULL
&& *cp
)
665 fprintf(obuf
, "%s=\"%s\"\n",
668 fprintf(obuf
, "%s\n", *p
);
672 if (obuf
!= stdout
) {
673 safe_signal(SIGPIPE
, SIG_IGN
);
675 safe_signal(SIGPIPE
, dflpipe
);
680 for (ap
= arglist
; *ap
!= NULL
; ap
++) {
683 varbuf
= ac_alloc(strlen(*ap
) + 1);
686 while (*cp
!= '=' && *cp
!= '\0')
693 if (strcmp(varbuf
, "") == 0) {
694 printf(tr(41, "Non-null variable name required\n"));
699 if (varbuf
[0] == 'n' && varbuf
[1] == 'o')
700 errs
+= unset_internal(&varbuf
[2]);
709 * Unset a bunch of variable values.
718 for (ap
= (char **)v
; *ap
!= NULL
; ap
++)
719 errs
+= unset_internal(*ap
);
724 * Put add users to a group.
730 struct grouphead
*gh
;
734 char **ap
, *gname
, **p
;
737 for (h
= 0, s
= 1; h
< HSHSIZE
; h
++)
738 for (gh
= groups
[h
]; gh
!= NULL
; gh
= gh
->g_link
)
741 ap
= (char **)salloc(s
* sizeof *ap
);
742 for (h
= 0, p
= ap
; h
< HSHSIZE
; h
++)
743 for (gh
= groups
[h
]; gh
!= NULL
; gh
= gh
->g_link
)
747 for (p
= ap
; *p
!= NULL
; p
++)
751 if (argv
[1] == NULL
) {
757 if ((gh
= findgroup(gname
)) == NULL
) {
758 gh
= (struct grouphead
*)scalloc(1, sizeof *gh
);
759 gh
->g_name
= vcopy(gname
);
761 gh
->g_link
= groups
[h
];
766 * Insert names from the command list into the group.
767 * Who cares if there are duplicates? They get tossed
771 for (ap
= argv
+1; *ap
!= NULL
; ap
++) {
772 gp
= (struct group
*)scalloc(1, sizeof *gp
);
773 gp
->ge_name
= vcopy(*ap
);
774 gp
->ge_link
= gh
->g_list
;
781 * Delete the passed groups.
789 printf(catgets(catd
, CATSET
, 209,
790 "Must specify alias or group to remove\n"));
795 while (*++argv
!= NULL
);
800 * Sort the passed string vecotor into ascending dictionary
808 for (ap
= list
; *ap
!= NULL
; ap
++)
812 qsort(list
, ap
-list
, sizeof(*list
), diction
);
816 * Do a dictionary order comparison of the arguments from
820 diction(const void *a
, const void *b
)
822 return(strcmp(*(char **)a
, *(char **)b
));
826 * Change to another file. With no argument, print information about
832 char **argv
= v
, *exp
;
834 if (argv
[0] == NULL
) {
838 if ((exp
= expand("&")) == NULL
)
840 strncpy(mboxname
, exp
, sizeof mboxname
)[sizeof mboxname
- 1] = '\0';
841 return (file1(*argv
));
850 fprintf(stderr
, "Cannot change folder from within a hook.\n");
853 i
= setfile(name
, 0);
856 callhook(mailname
, 0);
857 if (i
> 0 && value("emptystart") == NULL
)
859 announce(value("bsdcompat") != NULL
|| value("bsdannounce") != NULL
);
864 shellecho(const char *cp
)
904 while (n
-- && octalchar(cp
[1]&0377)) {
919 * Expand file names like echo
929 for (ap
= argv
; *ap
!= NULL
; ap
++) {
931 if ((cp
= expand(cp
)) != NULL
) {
934 cflag
|= shellecho(cp
);
945 return (respond_or_Respond('R'))((int *)v
, 0);
951 return (respond_or_Respond('R'))((int *)v
, 1);
955 * Reply to a series of messages by simply mailing to the senders
956 * and not messing around with the To: and Cc: lists as in normal
960 Respond_internal(int *msgvec
, int recipient_record
)
965 enum gfield gf
= value("fullnames") ? GFULL
: GSKIN
;
969 memset(&head
, 0, sizeof head
);
970 for (ap
= msgvec
; *ap
!= 0; ap
++) {
971 mp
= &message
[*ap
- 1];
974 if ((cp
= hfield1("reply-to", mp
)) == NULL
)
975 if ((cp
= hfield1("from", mp
)) == NULL
)
977 head
.h_to
= cat(head
.h_to
, lextract(cp
, GTO
|gf
));
979 if (head
.h_to
== NULL
)
981 mp
= &message
[msgvec
[0] - 1];
982 head
.h_subject
= hfield1("subject", mp
);
983 head
.h_subject
= reedit(head
.h_subject
);
984 make_ref_and_cs(mp
, &head
);
985 Eflag
= value("skipemptybody") != NULL
;
986 if (mail1(&head
, 1, mp
, NULL
, recipient_record
, 0, 0, Eflag
) == OKAY
&&
987 value("markanswered") && (mp
->m_flag
& MANSWERED
) == 0)
988 mp
->m_flag
|= MANSWER
|MANSWERED
;
993 * Conditional commands. These allow one to parameterize one's
994 * .mailrc and do some things if sending, others if receiving.
1003 printf(catgets(catd
, CATSET
, 42, "Illegal nested \"if\"\n"));
1022 printf(catgets(catd
, CATSET
, 43,
1023 "Unrecognized if-keyword: \"%s\"\n"), cp
);
1030 * Implement 'else'. This is pretty simple -- we just
1031 * flip over the conditional flag.
1041 printf(catgets(catd
, CATSET
, 44,
1042 "\"Else\" without matching \"if\"\n"));
1058 printf(catgets(catd
, CATSET
, 45,
1059 "Mail's idea of conditions is screwed up\n"));
1067 * End of if statement. Just set cond back to anything.
1076 printf(catgets(catd
, CATSET
, 46,
1077 "\"Endif\" without matching \"if\"\n"));
1085 * Set the list of alternate names.
1090 char **namelist
= v
;
1092 char **ap
, **ap2
, *cp
;
1094 c
= argcount(namelist
) + 1;
1098 for (ap
= altnames
; *ap
; ap
++)
1105 altnames
= scalloc(c
, sizeof (char *));
1106 for (ap
= namelist
, ap2
= altnames
; *ap
; ap
++, ap2
++) {
1107 cp
= scalloc(strlen(*ap
) + 1, sizeof (char));
1116 * Do the real work of resending.
1119 resend1(void *v
, int add_resent
)
1124 int f
, *ip
, *msgvec
;
1128 msgvec
= (int *)salloc((msgCount
+ 2) * sizeof *msgvec
);
1129 name
= laststring(str
, &f
, 1);
1131 puts(catgets(catd
, CATSET
, 47, "No recipient specified."));
1135 *msgvec
= first(0, MMNORM
);
1139 puts(catgets(catd
, CATSET
, 48,
1140 "No applicable messages."));
1144 } else if (getmsglist(str
, msgvec
, 0) < 0)
1149 printf("No applicable messages.\n");
1152 sn
= nalloc(name
, GTO
);
1154 for (ip
= msgvec
; *ip
&& ip
- msgvec
< msgCount
; ip
++) {
1155 if (resend_msg(&message
[*ip
- 1], to
, add_resent
) != OKAY
)
1162 * Resend a message list to a third person.
1167 return resend1(v
, 1);
1171 * Resend a message list to a third person without adding headers.
1176 return resend1(v
, 0);
1180 * 'newmail' or 'inc' command: Check for new mail without writing old
1190 if ((mb
.mb_type
!= MB_IMAP
|| imap_newmail(1)) &&
1191 (val
= setfile(mailname
, 1)) == 0) {
1193 setdot(&message
[mdot
- 1]);
1199 list_shortcuts(void)
1203 for (s
= shortcuts
; s
; s
= s
->sh_next
)
1204 printf("%s=%s\n", s
->sh_short
, s
->sh_long
);
1210 char **args
= (char **)v
;
1213 if (args
[0] == NULL
) {
1217 if (args
[1] == NULL
) {
1218 fprintf(stderr
, catgets(catd
, CATSET
, 220,
1219 "expansion name for shortcut missing\n"));
1222 if (args
[2] != NULL
) {
1223 fprintf(stderr
, catgets(catd
, CATSET
, 221,
1224 "too many arguments\n"));
1227 if ((s
= get_shortcut(args
[0])) != NULL
) {
1229 s
->sh_long
= sstrdup(args
[1]);
1231 s
= scalloc(1, sizeof *s
);
1232 s
->sh_short
= sstrdup(args
[0]);
1233 s
->sh_long
= sstrdup(args
[1]);
1234 s
->sh_next
= shortcuts
;
1241 get_shortcut(const char *str
)
1245 for (s
= shortcuts
; s
; s
= s
->sh_next
)
1246 if (strcmp(str
, s
->sh_short
) == 0)
1252 delete_shortcut(const char *str
)
1254 struct shortcut
*sp
, *sq
;
1256 for (sp
= shortcuts
, sq
= NULL
; sp
; sq
= sp
, sp
= sp
->sh_next
) {
1257 if (strcmp(sp
->sh_short
, str
) == 0) {
1261 sq
->sh_next
= sp
->sh_next
;
1262 if (sp
== shortcuts
)
1263 shortcuts
= sp
->sh_next
;
1274 char **args
= (char **)v
;
1277 if (args
[0] == NULL
) {
1278 fprintf(stderr
, catgets(catd
, CATSET
, 222,
1279 "need shortcut names to remove\n"));
1282 while (*args
!= NULL
) {
1283 if (delete_shortcut(*args
) != OKAY
) {
1285 fprintf(stderr
, catgets(catd
, CATSET
, 223,
1286 "%s: no such shortcut\n"), *args
);
1294 struct oldaccount
*ac_next
; /* next account in list */
1295 char *ac_name
; /* name of account */
1296 char **ac_vars
; /* variables to set */
1299 static struct oldaccount
*oldaccounts
;
1302 get_oldaccount(const char *name
)
1304 struct oldaccount
*a
;
1306 for (a
= oldaccounts
; a
; a
= a
->ac_next
)
1307 if (a
->ac_name
&& strcmp(name
, a
->ac_name
) == 0)
1315 char **args
= (char **)v
;
1316 struct oldaccount
*a
;
1318 int i
, mc
, oqf
, nqf
;
1321 if (args
[0] == NULL
) {
1322 if ((fp
= Ftemp(&cp
, "Ra", "w+", 0600, 1)) == NULL
) {
1328 mc
= listaccounts(fp
);
1329 for (a
= oldaccounts
; a
; a
= a
->ac_next
)
1333 fprintf(fp
, "%s:\n", a
->ac_name
);
1334 for (i
= 0; a
->ac_vars
[i
]; i
++)
1335 fprintf(fp
, "\t%s\n", a
->ac_vars
[i
]);
1342 if (args
[1] && args
[1][0] == '{' && args
[1][1] == '\0') {
1343 if (args
[2] != NULL
) {
1344 fprintf(stderr
, "Syntax is: account <name> {\n");
1347 if ((a
= get_oldaccount(args
[0])) != NULL
)
1349 return define1(args
[0], 1);
1352 if ((cp
= expand("&")) == NULL
)
1354 strncpy(mboxname
, cp
, sizeof mboxname
)[sizeof mboxname
- 1] = '\0';
1356 oqf
= savequitflags();
1357 if ((a
= get_oldaccount(args
[0])) == NULL
) {
1359 a
= scalloc(1, sizeof *a
);
1360 a
->ac_next
= oldaccounts
;
1363 if ((i
= callaccount(args
[0])) != CBAD
)
1365 printf("Account %s does not exist.\n", args
[0]);
1370 delaccount(args
[0]);
1371 a
->ac_name
= sstrdup(args
[0]);
1372 for (i
= 1; args
[i
]; i
++);
1373 a
->ac_vars
= scalloc(i
, sizeof *a
->ac_vars
);
1374 for (i
= 0; args
[i
+1]; i
++)
1375 a
->ac_vars
[i
] = sstrdup(args
[i
+1]);
1377 unset_allow_undefined
= 1;
1379 unset_allow_undefined
= 0;
1380 setf
: if (!starting
) {
1381 nqf
= savequitflags();
1382 restorequitflags(oqf
);
1384 restorequitflags(nqf
);
1398 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1399 m
= &message
[*ip
-1];
1401 if ((m
->m_flag
& (MFLAG
|MFLAGGED
)) == 0)
1402 m
->m_flag
|= MFLAG
|MFLAGGED
;
1414 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1415 m
= &message
[*ip
-1];
1417 if (m
->m_flag
& (MFLAG
|MFLAGGED
)) {
1418 m
->m_flag
&= ~(MFLAG
|MFLAGGED
);
1419 m
->m_flag
|= MUNFLAG
;
1432 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1433 m
= &message
[*ip
-1];
1435 if ((m
->m_flag
& (MANSWER
|MANSWERED
)) == 0)
1436 m
->m_flag
|= MANSWER
|MANSWERED
;
1442 cunanswered(void *v
)
1448 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1449 m
= &message
[*ip
-1];
1451 if (m
->m_flag
& (MANSWER
|MANSWERED
)) {
1452 m
->m_flag
&= ~(MANSWER
|MANSWERED
);
1453 m
->m_flag
|= MUNANSWER
;
1466 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1467 m
= &message
[*ip
-1];
1469 if ((m
->m_flag
& (MDRAFT
|MDRAFTED
)) == 0)
1470 m
->m_flag
|= MDRAFT
|MDRAFTED
;
1482 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1483 m
= &message
[*ip
-1];
1485 if (m
->m_flag
& (MDRAFT
|MDRAFTED
)) {
1486 m
->m_flag
&= ~(MDRAFT
|MDRAFTED
);
1487 m
->m_flag
|= MUNDRAFT
;
1498 * This is not perfect, but correct for machines with a 32-bit
1499 * IEEE float and a 32-bit unsigned long, and does at least not
1500 * produce SIGFPE on the Cray Y-MP.
1507 u
.l
= 0xff800000; /* -inf */
1509 #elif defined (INFINITY)
1511 #elif defined (HUGE_VALF)
1513 #elif defined (FLT_MAX)
1527 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1528 m
= &message
[*ip
-1];
1530 m
->m_score
= huge();
1542 for (ip
= msgvec
; *ip
!= 0; ip
++) {
1543 m
= &message
[*ip
-1];
1544 m
->m_flag
&= ~MKILL
;
1555 int f
, *msgvec
, *ip
;
1559 msgvec
= salloc((msgCount
+2) * sizeof *msgvec
);
1560 if ((sscore
= laststring(str
, &f
, 0)) == NULL
) {
1561 fprintf(stderr
, "No score given.\n");
1564 nscore
= strtod(sscore
, &xp
);
1566 fprintf(stderr
, "Invalid score: \"%s\"\n", sscore
);
1569 if (nscore
> FLT_MAX
)
1571 else if (nscore
< -FLT_MAX
)
1574 *msgvec
= first(0, MMNORM
);
1578 fprintf(stderr
, "No messages to score.\n");
1582 } else if (getmsglist(str
, msgvec
, 0) < 0)
1587 fprintf(stderr
, "No applicable messages.\n");
1590 for (ip
= msgvec
; *ip
&& ip
-msgvec
< msgCount
; ip
++) {
1591 m
= &message
[*ip
-1];
1592 if (m
->m_score
!= huge()) {
1593 m
->m_score
+= nscore
;
1596 else if (m
->m_score
> 0)
1597 m
->m_flag
&= ~MKILL
;
1598 if (m
->m_score
>= 0)
1611 switch (mb
.mb_type
) {
1632 if (*args
== NULL
) {
1633 fprintf(stderr
, tr(290, "Syntax is: remove mailbox ...\n"));
1637 if ((name
= expand(*args
)) == NULL
)
1639 if (strcmp(name
, mailname
) == 0) {
1640 fprintf(stderr
, tr(286,
1641 "Cannot remove current mailbox \"%s\".\n"),
1646 snprintf(vb
, sizeof vb
, tr(287, "Remove \"%s\" (y/n) ? "),
1650 switch (which_protocol(name
)) {
1652 if (unlink(name
) < 0) { /* do not handle .gz .bz2 */
1658 fprintf(stderr
, tr(288,
1659 "Cannot remove POP3 mailbox \"%s\".\n"),
1664 if (imap_remove(name
) != OKAY
)
1668 if (maildir_remove(name
) != OKAY
)
1672 fprintf(stderr
, tr(289,
1673 "Unknown protocol in \"%s\". Not removed.\n"),
1684 char **args
= v
, *old
, *new;
1685 enum protocol oldp
, newp
;
1688 if (args
[0] == NULL
|| args
[1] == NULL
|| args
[2] != NULL
) {
1689 fprintf(stderr
, "Syntax: rename old new\n");
1693 if ((old
= expand(args
[0])) == NULL
)
1695 oldp
= which_protocol(old
);
1696 if ((new = expand(args
[1])) == NULL
)
1698 newp
= which_protocol(new);
1700 if (strcmp(old
, mailname
) == 0 || strcmp(new, mailname
) == 0) {
1701 fprintf(stderr
, tr(291,
1702 "Cannot rename current mailbox \"%s\".\n"), old
);
1705 if ((oldp
== PROTO_IMAP
|| newp
== PROTO_IMAP
) && oldp
!= newp
) {
1706 fprintf(stderr
, tr(292,
1707 "Can only rename folders of same type.\n"));
1710 if (newp
== PROTO_POP3
)
1714 if (link(old
, new) < 0) {
1728 } else if (unlink(old
) < 0) {
1734 if (rename(old
, new) < 0) {
1740 nopop3
: fprintf(stderr
, tr(293, "Cannot rename POP3 mailboxes.\n"));
1744 if (imap_rename(old
, new) != OKAY
)
1748 fprintf(stderr
, tr(294,
1749 "Unknown protocol in \"%s\" and \"%s\". "
1750 "Not renamed.\n"), old
, new);