2023-03-07 20:43:04 -08:00
|
|
|
/* $OpenBSD: lex.c,v 1.43 2023/03/08 04:43:11 guenther Exp $ */
|
1997-07-13 14:21:08 -07:00
|
|
|
/* $NetBSD: lex.c,v 1.10 1997/05/17 19:55:13 pk Exp $ */
|
1996-03-27 11:32:19 -08:00
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1980, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
2003-06-02 19:56:05 -07:00
|
|
|
* 3. Neither the name of the University nor the names of its contributors
|
1995-10-18 01:37:01 -07:00
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "rcv.h"
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include "extern.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Mail -- a mail program
|
|
|
|
*
|
|
|
|
* Lexical processing of commands.
|
|
|
|
*/
|
|
|
|
|
|
|
|
char *prompt = "& ";
|
|
|
|
|
2001-01-15 21:36:08 -08:00
|
|
|
const struct cmd *com; /* command we are running */
|
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
|
|
|
* Set up editing on the given file name.
|
|
|
|
* If the first character of name is %, we are considered to be
|
|
|
|
* editing the file, otherwise we are reading our mail which has
|
|
|
|
* signficance for mbox and so forth.
|
|
|
|
*/
|
|
|
|
int
|
2001-11-21 07:26:39 -08:00
|
|
|
setfile(char *name)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
FILE *ibuf;
|
1997-07-24 09:23:36 -07:00
|
|
|
int i, fd;
|
1995-10-18 01:37:01 -07:00
|
|
|
struct stat stb;
|
|
|
|
char isedit = *name != '%';
|
2018-09-15 19:38:57 -07:00
|
|
|
const char *who = name[1] ? name + 1 : myname;
|
1997-07-24 10:27:09 -07:00
|
|
|
char tempname[PATHSIZE];
|
1995-10-18 01:37:01 -07:00
|
|
|
static int shudclob;
|
|
|
|
|
1997-07-13 17:24:24 -07:00
|
|
|
if ((name = expand(name)) == NULL)
|
1997-07-13 14:21:08 -07:00
|
|
|
return(-1);
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
if ((ibuf = Fopen(name, "r")) == NULL) {
|
|
|
|
if (!isedit && errno == ENOENT)
|
|
|
|
goto nomail;
|
2000-06-30 09:00:03 -07:00
|
|
|
warn("%s", name);
|
1995-10-18 01:37:01 -07:00
|
|
|
return(-1);
|
|
|
|
}
|
|
|
|
|
2019-06-28 06:34:58 -07:00
|
|
|
if (fstat(fileno(ibuf), &stb) == -1) {
|
1997-07-13 14:21:08 -07:00
|
|
|
warn("fstat");
|
|
|
|
(void)Fclose(ibuf);
|
|
|
|
return(-1);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (stb.st_mode & S_IFMT) {
|
|
|
|
case S_IFDIR:
|
1997-07-13 14:21:08 -07:00
|
|
|
(void)Fclose(ibuf);
|
2014-05-19 18:25:23 -07:00
|
|
|
warnc(EISDIR, "%s", name);
|
1997-07-13 14:21:08 -07:00
|
|
|
return(-1);
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
case S_IFREG:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
1997-07-13 14:21:08 -07:00
|
|
|
(void)Fclose(ibuf);
|
2014-05-19 18:25:23 -07:00
|
|
|
warnc(EINVAL, "%s", name);
|
1997-07-13 14:21:08 -07:00
|
|
|
return(-1);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Looks like all will be well. We must now relinquish our
|
|
|
|
* hold on the current set of stuff. Must hold signals
|
|
|
|
* while we are reading the new file, else we will ruin
|
|
|
|
* the message[] data structure.
|
|
|
|
*/
|
|
|
|
holdsigs();
|
|
|
|
if (shudclob)
|
|
|
|
quit();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copy the messages into /tmp
|
|
|
|
* and set pointers.
|
|
|
|
*/
|
|
|
|
readonly = 0;
|
For open/openat, if the flags parameter does not contain O_CREAT, the
3rd (variadic) mode_t parameter is irrelevant. Many developers in the past
have passed mode_t (0, 044, 0644, or such), which might lead future people
to copy this broken idiom, and perhaps even believe this parameter has some
meaning or implication or application. Delete them all.
This comes out of a conversation where tb@ noticed that a strange (but
intentional) pledge behaviour is to always knock-out high-bits from
mode_t on a number of system calls as a safety factor, and his bewilderment
that this appeared to be happening against valid modes (at least visually),
but no sorry, they are all irrelevant junk. They could all be 0xdeafbeef.
ok millert
2021-10-24 14:24:15 -07:00
|
|
|
if ((i = open(name, O_WRONLY)) == -1)
|
1995-10-18 01:37:01 -07:00
|
|
|
readonly++;
|
|
|
|
else
|
1997-07-13 14:21:08 -07:00
|
|
|
(void)close(i);
|
1995-10-18 01:37:01 -07:00
|
|
|
if (shudclob) {
|
1997-07-13 14:21:08 -07:00
|
|
|
(void)fclose(itf);
|
|
|
|
(void)fclose(otf);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
shudclob = 1;
|
|
|
|
edit = isedit;
|
2001-11-21 12:41:55 -08:00
|
|
|
strlcpy(prevfile, mailname, PATHSIZE);
|
2001-11-21 07:26:39 -08:00
|
|
|
if (name != mailname)
|
|
|
|
strlcpy(mailname, name, sizeof(mailname));
|
1995-10-18 01:37:01 -07:00
|
|
|
mailsize = fsize(ibuf);
|
1997-07-24 09:23:36 -07:00
|
|
|
(void)snprintf(tempname, sizeof(tempname),
|
|
|
|
"%s/mail.RxXXXXXXXXXX", tmpdir);
|
2014-10-26 13:38:13 -07:00
|
|
|
if ((fd = mkostemp(tempname, O_CLOEXEC)) == -1 ||
|
1997-07-24 09:23:36 -07:00
|
|
|
(otf = fdopen(fd, "w")) == NULL)
|
2000-06-30 09:00:03 -07:00
|
|
|
err(1, "%s", tempname);
|
2014-10-26 13:38:13 -07:00
|
|
|
if ((itf = fopen(tempname, "re")) == NULL)
|
2000-06-30 09:00:03 -07:00
|
|
|
err(1, "%s", tempname);
|
1997-07-24 10:27:09 -07:00
|
|
|
(void)rm(tempname);
|
2004-05-10 08:25:51 -07:00
|
|
|
setptr(ibuf, (off_t)0);
|
1995-10-18 01:37:01 -07:00
|
|
|
setmsize(msgCount);
|
1997-07-13 14:21:08 -07:00
|
|
|
/*
|
|
|
|
* New mail may have arrived while we were reading
|
|
|
|
* the mail file, so reset mailsize to be where
|
|
|
|
* we really are in the file...
|
|
|
|
*/
|
|
|
|
mailsize = ftell(ibuf);
|
|
|
|
(void)Fclose(ibuf);
|
1995-10-18 01:37:01 -07:00
|
|
|
relsesigs();
|
|
|
|
sawcom = 0;
|
|
|
|
if (!edit && msgCount == 0) {
|
|
|
|
nomail:
|
|
|
|
fprintf(stderr, "No mail for %s\n", who);
|
1997-07-13 14:21:08 -07:00
|
|
|
return(-1);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
1997-07-13 14:21:08 -07:00
|
|
|
/*
|
|
|
|
* Incorporate any new mail that has arrived since we first
|
|
|
|
* started reading mail.
|
|
|
|
*/
|
|
|
|
int
|
2001-11-21 07:26:39 -08:00
|
|
|
incfile(void)
|
1997-07-13 14:21:08 -07:00
|
|
|
{
|
|
|
|
int newsize;
|
|
|
|
int omsgCount = msgCount;
|
|
|
|
FILE *ibuf;
|
|
|
|
|
|
|
|
ibuf = Fopen(mailname, "r");
|
|
|
|
if (ibuf == NULL)
|
|
|
|
return(-1);
|
|
|
|
holdsigs();
|
2001-11-17 11:10:25 -08:00
|
|
|
if (!spool_lock()) {
|
|
|
|
(void)Fclose(ibuf);
|
|
|
|
relsesigs();
|
1997-07-22 12:13:25 -07:00
|
|
|
return(-1);
|
2001-11-17 11:10:25 -08:00
|
|
|
}
|
1997-07-13 14:21:08 -07:00
|
|
|
newsize = fsize(ibuf);
|
1997-07-21 23:46:20 -07:00
|
|
|
/* make sure mail box has grown and is non-empty */
|
|
|
|
if (newsize == 0 || newsize <= mailsize) {
|
2001-11-17 11:10:25 -08:00
|
|
|
(void)Fclose(ibuf);
|
1997-07-22 12:13:25 -07:00
|
|
|
spool_unlock();
|
1997-07-21 23:46:20 -07:00
|
|
|
relsesigs();
|
|
|
|
return(newsize == mailsize ? 0 : -1);
|
|
|
|
}
|
1997-07-13 14:21:08 -07:00
|
|
|
setptr(ibuf, mailsize);
|
|
|
|
setmsize(msgCount);
|
|
|
|
mailsize = ftell(ibuf);
|
|
|
|
(void)Fclose(ibuf);
|
1997-07-22 12:13:25 -07:00
|
|
|
spool_unlock();
|
1997-07-13 14:21:08 -07:00
|
|
|
relsesigs();
|
|
|
|
return(msgCount - omsgCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
int *msgvec;
|
2001-11-20 12:50:00 -08:00
|
|
|
int reset_on_stop; /* reset prompt if stopped */
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Interpret user commands one by one. If standard input is not a tty,
|
|
|
|
* print no prompt.
|
|
|
|
*/
|
|
|
|
void
|
2001-11-21 07:26:39 -08:00
|
|
|
commands(void)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2001-11-20 12:50:00 -08:00
|
|
|
int n, sig, *sigp;
|
|
|
|
int eofloop = 0;
|
1995-10-18 01:37:01 -07:00
|
|
|
char linebuf[LINESIZE];
|
|
|
|
|
2001-11-20 12:50:00 -08:00
|
|
|
prompt:
|
1995-10-18 01:37:01 -07:00
|
|
|
for (;;) {
|
|
|
|
/*
|
|
|
|
* Print the prompt, if needed. Clear out
|
|
|
|
* string space, and flush the output.
|
|
|
|
*/
|
1997-07-13 17:24:24 -07:00
|
|
|
if (!sourcing && value("interactive") != NULL) {
|
|
|
|
if ((value("autoinc") != NULL) && (incfile() > 0))
|
1997-07-13 14:21:08 -07:00
|
|
|
puts("New mail has arrived.");
|
1995-10-18 01:37:01 -07:00
|
|
|
reset_on_stop = 1;
|
2000-07-05 23:24:21 -07:00
|
|
|
printf("%s", prompt);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
sreset();
|
|
|
|
/*
|
|
|
|
* Read a line of commands from the current input
|
|
|
|
* and handle end of file specially.
|
|
|
|
*/
|
|
|
|
n = 0;
|
2001-11-20 12:50:00 -08:00
|
|
|
sig = 0;
|
|
|
|
sigp = sourcing ? NULL : &sig;
|
1995-10-18 01:37:01 -07:00
|
|
|
for (;;) {
|
2001-11-20 12:50:00 -08:00
|
|
|
if (readline(input, &linebuf[n], LINESIZE - n, sigp) < 0) {
|
|
|
|
if (sig) {
|
|
|
|
if (sig == SIGINT)
|
|
|
|
dointr();
|
|
|
|
else if (sig == SIGHUP)
|
|
|
|
/* nothing to do? */
|
|
|
|
exit(1);
|
|
|
|
else {
|
|
|
|
/* Stopped by job control */
|
|
|
|
(void)kill(0, sig);
|
|
|
|
if (reset_on_stop)
|
|
|
|
reset_on_stop = 0;
|
|
|
|
}
|
|
|
|
goto prompt;
|
|
|
|
}
|
1995-10-18 01:37:01 -07:00
|
|
|
if (n == 0)
|
|
|
|
n = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ((n = strlen(linebuf)) == 0)
|
|
|
|
break;
|
|
|
|
n--;
|
|
|
|
if (linebuf[n] != '\\')
|
|
|
|
break;
|
|
|
|
linebuf[n++] = ' ';
|
|
|
|
}
|
|
|
|
reset_on_stop = 0;
|
|
|
|
if (n < 0) {
|
|
|
|
/* eof */
|
|
|
|
if (loading)
|
|
|
|
break;
|
|
|
|
if (sourcing) {
|
|
|
|
unstack();
|
|
|
|
continue;
|
|
|
|
}
|
1997-07-13 17:24:24 -07:00
|
|
|
if (value("interactive") != NULL &&
|
|
|
|
value("ignoreeof") != NULL &&
|
1995-10-18 01:37:01 -07:00
|
|
|
++eofloop < 25) {
|
1997-07-13 14:21:08 -07:00
|
|
|
puts("Use \"quit\" to quit.");
|
1995-10-18 01:37:01 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
eofloop = 0;
|
|
|
|
if (execute(linebuf, 0))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Execute a single command.
|
|
|
|
* Command functions return 0 for success, 1 for error, and -1
|
|
|
|
* for abort. A 1 or -1 aborts a load or source. A -1 aborts
|
|
|
|
* the interactive command loop.
|
|
|
|
* Contxt is non-zero if called while composing mail.
|
|
|
|
*/
|
|
|
|
int
|
2001-11-21 07:26:39 -08:00
|
|
|
execute(char *linebuf, int contxt)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
char word[LINESIZE];
|
|
|
|
char *arglist[MAXARGC];
|
1997-11-13 16:23:41 -08:00
|
|
|
char *cp, *cp2;
|
|
|
|
int c, muvec[2];
|
1995-10-18 01:37:01 -07:00
|
|
|
int e = 1;
|
|
|
|
|
2001-01-15 21:36:08 -08:00
|
|
|
com = NULL;
|
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
|
|
|
* Strip the white space away from the beginning
|
|
|
|
* of the command, then scan out a word, which
|
|
|
|
* consists of anything except digits and white space.
|
|
|
|
*
|
|
|
|
* Handle ! escapes differently to get the correct
|
|
|
|
* lexical conventions.
|
|
|
|
*/
|
2014-01-17 10:42:30 -08:00
|
|
|
for (cp = linebuf; isspace((unsigned char)*cp); cp++)
|
1995-10-18 01:37:01 -07:00
|
|
|
;
|
|
|
|
if (*cp == '!') {
|
|
|
|
if (sourcing) {
|
1997-07-13 14:21:08 -07:00
|
|
|
puts("Can't \"!\" while sourcing");
|
1995-10-18 01:37:01 -07:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
shell(cp+1);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
cp2 = word;
|
2014-01-17 10:42:30 -08:00
|
|
|
while (*cp &&
|
|
|
|
strchr(" \t0123456789$^.:/-+*'\"", (unsigned char)*cp) == NULL)
|
1995-10-18 01:37:01 -07:00
|
|
|
*cp2++ = *cp++;
|
|
|
|
*cp2 = '\0';
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Look up the command; if not found, bitch.
|
|
|
|
* Normally, a blank command would map to the
|
|
|
|
* first command in the table; while sourcing,
|
|
|
|
* however, we ignore blank lines to eliminate
|
|
|
|
* confusion.
|
|
|
|
*/
|
|
|
|
if (sourcing && *word == '\0')
|
|
|
|
return(0);
|
|
|
|
com = lex(word);
|
2001-11-21 07:26:39 -08:00
|
|
|
if (com == NULL) {
|
1995-10-18 01:37:01 -07:00
|
|
|
printf("Unknown command: \"%s\"\n", word);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See if we should execute the command -- if a conditional
|
|
|
|
* we always execute it, otherwise, check the state of cond.
|
|
|
|
*/
|
|
|
|
if ((com->c_argtype & F) == 0)
|
1996-06-11 05:53:31 -07:00
|
|
|
if ((cond == CRCV && !rcvmode) || (cond == CSEND && rcvmode))
|
1995-10-18 01:37:01 -07:00
|
|
|
return(0);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Process the arguments to the command, depending
|
|
|
|
* on the type he expects. Default to an error.
|
|
|
|
* If we are sourcing an interactive command, it's
|
|
|
|
* an error.
|
|
|
|
*/
|
|
|
|
if (!rcvmode && (com->c_argtype & M) == 0) {
|
|
|
|
printf("May not execute \"%s\" while sending\n",
|
|
|
|
com->c_name);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (sourcing && com->c_argtype & I) {
|
|
|
|
printf("May not execute \"%s\" while sourcing\n",
|
|
|
|
com->c_name);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (readonly && com->c_argtype & W) {
|
|
|
|
printf("May not execute \"%s\" -- message file is read only\n",
|
|
|
|
com->c_name);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (contxt && com->c_argtype & R) {
|
|
|
|
printf("Cannot recursively invoke \"%s\"\n", com->c_name);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
switch (com->c_argtype & ~(F|P|I|M|T|W|R)) {
|
2001-01-15 21:36:08 -08:00
|
|
|
case MSGLIST|STRLIST:
|
|
|
|
/*
|
|
|
|
* A message list defaulting to nearest forward
|
|
|
|
* legal message.
|
|
|
|
*/
|
|
|
|
if (msgvec == 0) {
|
|
|
|
puts("Illegal use of \"message list\"");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* remove leading blanks.
|
|
|
|
*/
|
2014-01-17 10:42:30 -08:00
|
|
|
while (isspace((unsigned char)*cp))
|
2001-01-15 21:36:08 -08:00
|
|
|
cp++;
|
|
|
|
|
2014-01-17 10:42:30 -08:00
|
|
|
if (isdigit((unsigned char)*cp) || *cp == ':') {
|
2001-01-15 21:36:08 -08:00
|
|
|
if ((c = getmsglist(cp, msgvec, com->c_msgflag)) < 0)
|
|
|
|
break;
|
|
|
|
/* position to next space - past the message list */
|
2014-01-17 10:42:30 -08:00
|
|
|
while (!isspace((unsigned char)*cp))
|
2001-01-15 21:36:08 -08:00
|
|
|
cp++;
|
|
|
|
/* position to next non-space */
|
2014-01-17 10:42:30 -08:00
|
|
|
while (isspace((unsigned char)*cp))
|
2001-01-15 21:36:08 -08:00
|
|
|
cp++;
|
|
|
|
} else {
|
|
|
|
c = 0; /* no message list */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == 0) {
|
|
|
|
*msgvec = first(com->c_msgflag,
|
|
|
|
com->c_msgmask);
|
2011-04-06 04:36:23 -07:00
|
|
|
msgvec[1] = 0;
|
2001-01-15 21:36:08 -08:00
|
|
|
}
|
2003-10-24 13:30:02 -07:00
|
|
|
if (*msgvec == 0) {
|
2001-01-15 21:36:08 -08:00
|
|
|
puts("No applicable messages");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Just the straight string, with
|
|
|
|
* leading blanks removed.
|
|
|
|
*/
|
2014-01-17 10:42:30 -08:00
|
|
|
while (isspace((unsigned char)*cp))
|
2001-01-15 21:36:08 -08:00
|
|
|
cp++;
|
|
|
|
|
|
|
|
e = (*com->c_func2)(msgvec, cp);
|
|
|
|
break;
|
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
case MSGLIST:
|
|
|
|
/*
|
|
|
|
* A message list defaulting to nearest forward
|
|
|
|
* legal message.
|
|
|
|
*/
|
2004-05-10 05:10:50 -07:00
|
|
|
if (msgvec == NULL) {
|
1997-07-13 14:21:08 -07:00
|
|
|
puts("Illegal use of \"message list\"");
|
1995-10-18 01:37:01 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ((c = getmsglist(cp, msgvec, com->c_msgflag)) < 0)
|
|
|
|
break;
|
|
|
|
if (c == 0) {
|
|
|
|
*msgvec = first(com->c_msgflag,
|
|
|
|
com->c_msgmask);
|
2011-04-06 04:36:23 -07:00
|
|
|
msgvec[1] = 0;
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
2003-10-24 13:30:02 -07:00
|
|
|
if (*msgvec == 0) {
|
1997-07-13 14:21:08 -07:00
|
|
|
puts("No applicable messages");
|
1995-10-18 01:37:01 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
e = (*com->c_func)(msgvec);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NDMLIST:
|
|
|
|
/*
|
|
|
|
* A message list with no defaults, but no error
|
|
|
|
* if none exist.
|
|
|
|
*/
|
|
|
|
if (msgvec == 0) {
|
1997-07-13 14:21:08 -07:00
|
|
|
puts("Illegal use of \"message list\"");
|
1995-10-18 01:37:01 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (getmsglist(cp, msgvec, com->c_msgflag) < 0)
|
|
|
|
break;
|
|
|
|
e = (*com->c_func)(msgvec);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case STRLIST:
|
|
|
|
/*
|
|
|
|
* Just the straight string, with
|
|
|
|
* leading blanks removed.
|
|
|
|
*/
|
2014-01-17 10:42:30 -08:00
|
|
|
while (isspace((unsigned char)*cp))
|
1995-10-18 01:37:01 -07:00
|
|
|
cp++;
|
|
|
|
e = (*com->c_func)(cp);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RAWLIST:
|
|
|
|
/*
|
|
|
|
* A vector of strings, in shell style.
|
|
|
|
*/
|
|
|
|
if ((c = getrawlist(cp, arglist,
|
1997-07-13 14:21:08 -07:00
|
|
|
sizeof(arglist) / sizeof(*arglist))) < 0)
|
1995-10-18 01:37:01 -07:00
|
|
|
break;
|
|
|
|
if (c < com->c_minargs) {
|
|
|
|
printf("%s requires at least %d arg(s)\n",
|
|
|
|
com->c_name, com->c_minargs);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (c > com->c_maxargs) {
|
|
|
|
printf("%s takes no more than %d arg(s)\n",
|
|
|
|
com->c_name, com->c_maxargs);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
e = (*com->c_func)(arglist);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NOLIST:
|
|
|
|
/*
|
|
|
|
* Just the constant zero, for exiting,
|
|
|
|
* eg.
|
|
|
|
*/
|
|
|
|
e = (*com->c_func)(0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
1997-11-13 16:23:41 -08:00
|
|
|
errx(1, "Unknown argtype");
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
/*
|
|
|
|
* Exit the current source file on
|
|
|
|
* error.
|
|
|
|
*/
|
|
|
|
if (e) {
|
|
|
|
if (e < 0)
|
1997-07-13 14:21:08 -07:00
|
|
|
return(1);
|
1995-10-18 01:37:01 -07:00
|
|
|
if (loading)
|
1997-07-13 14:21:08 -07:00
|
|
|
return(1);
|
1995-10-18 01:37:01 -07:00
|
|
|
if (sourcing)
|
|
|
|
unstack();
|
1997-07-13 14:21:08 -07:00
|
|
|
return(0);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
1996-06-11 05:53:31 -07:00
|
|
|
if (com == NULL)
|
|
|
|
return(0);
|
1997-07-13 17:24:24 -07:00
|
|
|
if (value("autoprint") != NULL && com->c_argtype & P)
|
1995-10-18 01:37:01 -07:00
|
|
|
if ((dot->m_flag & MDELETED) == 0) {
|
|
|
|
muvec[0] = dot - &message[0] + 1;
|
|
|
|
muvec[1] = 0;
|
|
|
|
type(muvec);
|
|
|
|
}
|
|
|
|
if (!sourcing && (com->c_argtype & T) == 0)
|
|
|
|
sawcom = 1;
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the size of the message vector used to construct argument
|
|
|
|
* lists to message list functions.
|
|
|
|
*/
|
|
|
|
void
|
2001-11-21 12:41:55 -08:00
|
|
|
setmsize(int n)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2003-10-12 17:46:08 -07:00
|
|
|
int *msgvec2;
|
2001-11-21 12:41:55 -08:00
|
|
|
size_t msize;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
2001-11-21 12:41:55 -08:00
|
|
|
msize = (n + 1) * sizeof(*msgvec);
|
2003-10-12 17:46:08 -07:00
|
|
|
if ((msgvec2 = realloc(msgvec, msize)) == NULL)
|
2015-10-16 10:56:07 -07:00
|
|
|
err(1, "realloc");
|
2003-10-12 17:46:08 -07:00
|
|
|
msgvec = msgvec2;
|
2001-11-21 12:41:55 -08:00
|
|
|
memset(msgvec, 0, msize);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find the correct command in the command table corresponding
|
|
|
|
* to the passed command "word"
|
|
|
|
*/
|
|
|
|
|
1996-03-27 11:32:19 -08:00
|
|
|
const struct cmd *
|
2001-11-21 07:26:39 -08:00
|
|
|
lex(char *word)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
1996-03-27 11:32:19 -08:00
|
|
|
extern const struct cmd cmdtab[];
|
1997-11-13 16:23:41 -08:00
|
|
|
const struct cmd *cp;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
1998-09-08 07:59:12 -07:00
|
|
|
if (word[0] == '#')
|
|
|
|
word = "#";
|
1997-07-13 17:24:24 -07:00
|
|
|
for (cp = &cmdtab[0]; cp->c_name != NULL; cp++)
|
1995-10-18 01:37:01 -07:00
|
|
|
if (isprefix(word, cp->c_name))
|
|
|
|
return(cp);
|
2001-11-21 07:26:39 -08:00
|
|
|
return(NULL);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Determine if as1 is a valid prefix of as2.
|
|
|
|
* Return true if yep.
|
|
|
|
*/
|
|
|
|
int
|
2001-11-21 07:26:39 -08:00
|
|
|
isprefix(char *as1, char *as2)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
1997-11-13 16:23:41 -08:00
|
|
|
char *s1, *s2;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
s1 = as1;
|
|
|
|
s2 = as2;
|
|
|
|
while (*s1++ == *s2)
|
|
|
|
if (*s2++ == '\0')
|
|
|
|
return(1);
|
|
|
|
return(*--s1 == '\0');
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The following gets called on receipt of an interrupt. This is
|
|
|
|
* to abort printout of a command, mainly.
|
|
|
|
* Dispatching here when command() is inactive crashes rcv.
|
|
|
|
* Close all open files except 0, 1, 2, and the temporary.
|
|
|
|
* Also, unstack all source files.
|
|
|
|
*/
|
|
|
|
int inithdr; /* am printing startup headers */
|
|
|
|
|
|
|
|
void
|
2001-11-21 07:26:39 -08:00
|
|
|
dointr(void)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
|
|
|
|
noreset = 0;
|
|
|
|
if (!inithdr)
|
|
|
|
sawcom++;
|
|
|
|
inithdr = 0;
|
|
|
|
while (sourcing)
|
|
|
|
unstack();
|
|
|
|
|
|
|
|
close_all_files();
|
|
|
|
|
|
|
|
if (image >= 0) {
|
1997-07-13 14:21:08 -07:00
|
|
|
(void)close(image);
|
1995-10-18 01:37:01 -07:00
|
|
|
image = -1;
|
|
|
|
}
|
1997-07-13 14:21:08 -07:00
|
|
|
fputs("Interrupt\n", stderr);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Announce the presence of the current Mail version,
|
|
|
|
* give the message count, and print a header listing.
|
|
|
|
*/
|
|
|
|
void
|
2001-11-21 07:26:39 -08:00
|
|
|
announce(void)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
int vec[2], mdot;
|
|
|
|
|
1997-07-13 14:21:08 -07:00
|
|
|
mdot = newfileinfo(0);
|
1995-10-18 01:37:01 -07:00
|
|
|
vec[0] = mdot;
|
|
|
|
vec[1] = 0;
|
|
|
|
dot = &message[mdot - 1];
|
1997-07-13 17:24:24 -07:00
|
|
|
if (msgCount > 0 && value("noheader") == NULL) {
|
1995-10-18 01:37:01 -07:00
|
|
|
inithdr++;
|
|
|
|
headers(vec);
|
|
|
|
inithdr = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Announce information about the file we are editing.
|
|
|
|
* Return a likely place to set dot.
|
|
|
|
*/
|
|
|
|
int
|
2001-11-21 07:26:39 -08:00
|
|
|
newfileinfo(int omsgCount)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
1997-11-13 16:23:41 -08:00
|
|
|
struct message *mp;
|
|
|
|
int u, n, mdot, d, s;
|
1997-07-13 14:21:08 -07:00
|
|
|
char fname[PATHSIZE], zname[PATHSIZE], *ename;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
1997-07-13 14:21:08 -07:00
|
|
|
for (mp = &message[omsgCount]; mp < &message[msgCount]; mp++)
|
1995-10-18 01:37:01 -07:00
|
|
|
if (mp->m_flag & MNEW)
|
|
|
|
break;
|
|
|
|
if (mp >= &message[msgCount])
|
1997-07-13 14:21:08 -07:00
|
|
|
for (mp = &message[omsgCount]; mp < &message[msgCount]; mp++)
|
1995-10-18 01:37:01 -07:00
|
|
|
if ((mp->m_flag & MREAD) == 0)
|
|
|
|
break;
|
|
|
|
if (mp < &message[msgCount])
|
|
|
|
mdot = mp - &message[0] + 1;
|
|
|
|
else
|
1997-07-13 14:21:08 -07:00
|
|
|
mdot = omsgCount + 1;
|
1995-10-18 01:37:01 -07:00
|
|
|
s = d = 0;
|
|
|
|
for (mp = &message[0], n = 0, u = 0; mp < &message[msgCount]; mp++) {
|
|
|
|
if (mp->m_flag & MNEW)
|
|
|
|
n++;
|
|
|
|
if ((mp->m_flag & MREAD) == 0)
|
|
|
|
u++;
|
|
|
|
if (mp->m_flag & MDELETED)
|
|
|
|
d++;
|
|
|
|
if (mp->m_flag & MSAVED)
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
ename = mailname;
|
1997-07-13 14:21:08 -07:00
|
|
|
if (getfold(fname, sizeof(fname)) >= 0) {
|
2001-11-21 07:26:39 -08:00
|
|
|
strlcat(fname, "/", sizeof(fname));
|
1995-10-18 01:37:01 -07:00
|
|
|
if (strncmp(fname, mailname, strlen(fname)) == 0) {
|
1997-07-24 10:27:09 -07:00
|
|
|
(void)snprintf(zname, sizeof(zname), "+%s",
|
1997-07-13 14:21:08 -07:00
|
|
|
mailname + strlen(fname));
|
1995-10-18 01:37:01 -07:00
|
|
|
ename = zname;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("\"%s\": ", ename);
|
|
|
|
if (msgCount == 1)
|
1997-07-13 14:21:08 -07:00
|
|
|
fputs("1 message", stdout);
|
1995-10-18 01:37:01 -07:00
|
|
|
else
|
|
|
|
printf("%d messages", msgCount);
|
|
|
|
if (n > 0)
|
|
|
|
printf(" %d new", n);
|
|
|
|
if (u-n > 0)
|
|
|
|
printf(" %d unread", u);
|
|
|
|
if (d > 0)
|
|
|
|
printf(" %d deleted", d);
|
|
|
|
if (s > 0)
|
|
|
|
printf(" %d saved", s);
|
|
|
|
if (readonly)
|
1997-07-13 14:21:08 -07:00
|
|
|
fputs(" [Read only]", stdout);
|
|
|
|
putchar('\n');
|
1995-10-18 01:37:01 -07:00
|
|
|
return(mdot);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Print the current version number.
|
|
|
|
*/
|
|
|
|
int
|
2001-11-21 07:26:39 -08:00
|
|
|
pversion(void *v)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2001-11-21 07:26:39 -08:00
|
|
|
extern const char version[];
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
printf("Version %s\n", version);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Load a file of user definitions.
|
|
|
|
*/
|
|
|
|
void
|
2001-11-21 07:26:39 -08:00
|
|
|
load(char *name)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
1997-11-13 16:23:41 -08:00
|
|
|
FILE *in, *oldin;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
if ((in = Fopen(name, "r")) == NULL)
|
|
|
|
return;
|
|
|
|
oldin = input;
|
|
|
|
input = in;
|
|
|
|
loading = 1;
|
|
|
|
sourcing = 1;
|
|
|
|
commands();
|
|
|
|
loading = 0;
|
|
|
|
sourcing = 0;
|
|
|
|
input = oldin;
|
1997-07-13 14:21:08 -07:00
|
|
|
(void)Fclose(in);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|