cmd3.c revision 7c478bd95313f5f23a4c958a745db2134aa03244
/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License, Version 1.0 only
* (the "License"). You may not use this file except in compliance
* with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
/* All Rights Reserved */
/*
* Copyright 1985-2002 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
/*
* University Copyright- Copyright (c) 1982, 1986, 1988
* The Regents of the University of California
* All Rights Reserved
*
* University Acknowledgment- Portions of this document are derived from
* software developed by the University of California, Berkeley, and its
* contributors.
*/
#pragma ident "%Z%%M% %I% %E% SMI"
#include "rcv.h"
#include <locale.h>
/*
* mailx -- a modified version of a University of California at Berkeley
* mail program
*
* Still more user commands.
*/
static int bangexp(char *str);
static int diction(const void *a, const void *b);
static char *getfilename(char *name, int *aedit);
static int resp1(int *msgvec, int useauthor);
static int Resp1(int *msgvec, int useauthor);
static char *reedit(char *subj);
static int shell1(char *str);
static void sort(char **list);
static char *replyto(struct message *mp, char **f);
static int reply2sender(void);
static char prevfile[PATHSIZE];
static char origprevfile[PATHSIZE];
static char lastbang[BUFSIZ];
/*
* Process a shell escape by saving signals, ignoring signals,
* and forking a sh -c
*/
int
shell(char *str)
{
shell1(str);
printf("!\n");
return(0);
}
static int
shell1(char *str)
{
void (*sig[2])(int);
register int t;
register pid_t p;
char *Shell;
char cmd[BUFSIZ];
nstrcpy(cmd, sizeof (cmd), str);
if (bangexp(cmd) < 0)
return(-1);
if ((Shell = value("SHELL")) == NOSTR || *Shell=='\0')
Shell = SHELL;
for (t = SIGINT; t <= SIGQUIT; t++)
sig[t-SIGINT] = sigset(t, SIG_IGN);
p = vfork();
if (p == 0) {
setuid(getuid());
sigchild();
for (t = SIGINT; t <= SIGQUIT; t++)
if (sig[t-SIGINT] != SIG_IGN)
sigsys(t, SIG_DFL);
execlp(Shell, Shell, "-c", cmd, (char *)0);
perror(Shell);
_exit(1);
}
while (wait(0) != p)
;
if (p == (pid_t)-1)
perror("fork");
for (t = SIGINT; t <= SIGQUIT; t++)
sigset(t, sig[t-SIGINT]);
return(0);
}
/*
* Fork an interactive shell.
*/
int
#ifdef __cplusplus
dosh(char *)
#else
/* ARGSUSED */
dosh(char *s)
#endif
{
void (*sig[2])(int);
register int t;
register pid_t p;
char *Shell;
if ((Shell = value("SHELL")) == NOSTR || *Shell=='\0')
Shell = SHELL;
for (t = SIGINT; t <= SIGQUIT; t++)
sig[t-SIGINT] = sigset(t, SIG_IGN);
p = vfork();
if (p == 0) {
setuid(getuid());
sigchild();
for (t = SIGINT; t <= SIGQUIT; t++)
if (sig[t-SIGINT] != SIG_IGN)
sigset(t, SIG_DFL);
execlp(Shell, Shell, (char *)0);
perror(Shell);
_exit(1);
}
while (wait(0) != p)
;
if (p == (pid_t)-1)
perror("fork");
for (t = SIGINT; t <= SIGQUIT; t++)
sigset(t, sig[t-SIGINT]);
putchar('\n');
return(0);
}
/*
* Expand the shell escape by expanding unescaped !'s into the
* last issued command where possible.
*/
static int
bangexp(char *str)
{
char bangbuf[BUFSIZ];
register char *cp, *cp2;
register int n;
int changed = 0;
int bangit = (value("bang")!=NOSTR);
cp = str;
cp2 = bangbuf;
n = BUFSIZ;
while (*cp) {
if (*cp=='!' && bangit) {
if (n < (int)strlen(lastbang)) {
overf:
printf(gettext("Command buffer overflow\n"));
return(-1);
}
changed++;
strcpy(cp2, lastbang);
cp2 += strlen(lastbang);
n -= strlen(lastbang);
cp++;
continue;
}
if (*cp == '\\' && cp[1] == '!') {
if (--n <= 1)
goto overf;
*cp2++ = '!';
cp += 2;
changed++;
}
if (--n <= 1)
goto overf;
*cp2++ = *cp++;
}
*cp2 = 0;
if (changed) {
printf("!%s\n", bangbuf);
fflush(stdout);
}
nstrcpy(str, BUFSIZ, bangbuf);
nstrcpy(lastbang, sizeof (lastbang), bangbuf);
return(0);
}
/*
* Print out a nice help message from some file or another.
*/
int
help(void)
{
register c;
register FILE *f;
if ((f = fopen(HELPFILE, "r")) == NULL) {
printf(gettext("No help just now.\n"));
return(1);
}
while ((c = getc(f)) != EOF)
putchar(c);
fclose(f);
return(0);
}
/*
* Change user's working directory.
*/
int
schdir(char *str)
{
register char *cp;
char cwd[PATHSIZE], file[PATHSIZE];
static char efile[PATHSIZE];
for (cp = str; *cp == ' '; cp++)
;
if (*cp == '\0')
cp = homedir;
else
if ((cp = expand(cp)) == NOSTR)
return(1);
if (editfile != NOSTR && (*editfile != '/' || mailname[0] != '/')) {
if (getcwd(cwd, (int)sizeof (cwd)) == 0) {
fprintf(stderr,
gettext("Can't get current directory: %s\n"), cwd);
return(1);
}
}
if (chdir(cp) < 0) {
perror(cp);
return(1);
}
/*
* Convert previously relative names to absolute names.
*/
if (editfile != NOSTR && *editfile != '/') {
snprintf(file, sizeof (file), "%s/%s", cwd, editfile);
nstrcpy(efile, sizeof (efile), file);
editfile = efile;
}
if (mailname[0] != '/') {
snprintf(file, sizeof (file), "%s/%s", cwd, mailname);
nstrcpy(mailname, PATHSIZE, file);
}
return(0);
}
/*
* Two versions of reply. Reply to all names in message or reply
* to only sender of message, depending on setting of "replyall".
*/
int
respond(int *msgvec)
{
if (reply2sender())
return(resp1(msgvec, 0));
else
return(Resp1(msgvec, 0));
}
int
followup(int *msgvec)
{
if (reply2sender())
return(resp1(msgvec, 1));
else
return(Resp1(msgvec, 1));
}
int
replyall(int *msgvec)
{
return(resp1(msgvec, 0));
}
static int
resp1(int *msgvec, int useauthor)
{
struct message *mp;
char *cp, *buf, *rcv, *skin_rcv, *reply2, **ap, *returnaddr;
struct name *np;
struct header head;
char mylocalname[BUFSIZ], mydomname[BUFSIZ];
if (msgvec[1] != 0) {
printf(gettext(
"Sorry, can't reply to multiple messages at once\n"));
return(1);
}
snprintf(mydomname, sizeof (mydomname), "%s@%s", myname, domain);
snprintf(mylocalname, sizeof (mylocalname), "%s@%s", myname, host);
returnaddr = value("returnaddr");
mp = &message[msgvec[0] - 1];
dot = mp;
reply2 = replyto(mp, &rcv);
cp = skin(hfield("to", mp, addto));
if (cp != NOSTR) {
buf = (char *)salloc(strlen(reply2) + strlen(cp) + 2);
strcpy(buf, reply2);
strcat(buf, " ");
strcat(buf, cp);
} else
buf = reply2;
np = elide(extract(buf, GTO));
#ifdef OPTIM
/* rcv = netrename(rcv); */
#endif /* OPTIM */
/*
* Delete my name from the reply list,
* and with it, all my alternate names.
*/
skin_rcv = skin(rcv);
mapf(np, skin_rcv);
np = delname(np, myname);
np = delname(np, mylocalname);
np = delname(np, mydomname);
if (returnaddr && *returnaddr)
np = delname(np, returnaddr);
if (altnames != 0)
for (ap = altnames; *ap; ap++)
np = delname(np, *ap);
head.h_seq = 1;
cp = detract(np, 0);
if (cp == NOSTR) {
if (reply2)
cp = unuucp(reply2);
else
cp = unuucp(rcv);
}
head.h_to = cp;
head.h_subject = hfield("subject", mp, addone);
if (head.h_subject == NOSTR)
head.h_subject = hfield("subj", mp, addone);
head.h_subject = reedit(head.h_subject);
head.h_cc = NOSTR;
cp = skin(hfield("cc", mp, addto));
if (cp != NOSTR) {
np = elide(extract(cp, GCC));
mapf(np, skin_rcv);
np = delname(np, myname);
np = delname(np, mylocalname);
np = delname(np, mydomname);
if (returnaddr && *returnaddr)
np = delname(np, returnaddr);
np = delname(np, skin_rcv);
if (altnames != 0)
for (ap = altnames; *ap; ap++)
np = delname(np, *ap);
head.h_cc = detract(np, 0);
}
head.h_bcc = NOSTR;
head.h_defopt = NOSTR;
head.h_others = NOSTRPTR;
mail1(&head, useauthor, useauthor ? rcv : NOSTR);
return(0);
}
void
getrecf(char *buf, char *recfile, int useauthor, int sz_recfile)
{
register char *bp, *cp;
register char *recf = recfile;
register int folderize;
char fldr[BUFSIZ];
folderize = (value("outfolder")!=NOSTR && getfold(fldr) == 0);
if (useauthor) {
if (folderize)
*recf++ = '+';
if (debug) fprintf(stderr, "buf='%s'\n", buf);
for (bp=skin(buf), cp=recf; *bp && !any(*bp, ", "); bp++) {
if (*bp=='!')
cp = recf;
else
*cp++ = *bp;
if (cp >= &recfile[sz_recfile - 1]) {
printf(gettext("File name buffer overflow\n"));
break;
}
}
*cp = '\0';
if (cp==recf)
*recfile = '\0';
/* now strip off any Internet host names */
if ((cp = strchr(recf, '%')) == NOSTR)
cp = strchr(recf, '@');
if (cp != NOSTR)
*cp = '\0';
} else {
if (cp = value("record")) {
int sz = PATHSIZE;
if (folderize && *cp!='+' && *cp!='/'
&& *safeexpand(cp)!='/') {
*recf++ = '+';
sz--;
}
nstrcpy(recf, sz, cp);
} else
*recf = '\0';
}
if (debug) fprintf(stderr, "recfile='%s'\n", recfile);
}
/*
* Modify the subject we are replying to to begin with Re: if
* it does not already.
*/
static char *
reedit(char *subj)
{
char sbuf[10];
register char *newsubj;
if (subj == NOSTR)
return(NOSTR);
strncpy(sbuf, subj, 3);
sbuf[3] = 0;
if (icequal(sbuf, "re:"))
return(subj);
newsubj = (char *)salloc((unsigned)(strlen(subj) + 5));
sprintf(newsubj, "Re: %s", subj);
return(newsubj);
}
/*
* Preserve the named messages, so that they will be sent
* back to the system mailbox.
*/
int
preserve(int *msgvec)
{
register struct message *mp;
register int *ip, mesg;
if (edit) {
printf(gettext("Cannot \"preserve\" in edit mode\n"));
return(1);
}
for (ip = msgvec; *ip != NULL; ip++) {
mesg = *ip;
mp = &message[mesg-1];
mp->m_flag |= MPRESERVE;
mp->m_flag &= ~MBOX;
dot = mp;
}
return(0);
}
/*
* Mark all given messages as unread.
*/
int
unread(int msgvec[])
{
register int *ip;
for (ip = msgvec; *ip != NULL; ip++) {
dot = &message[*ip-1];
dot->m_flag &= ~(MREAD|MTOUCH);
dot->m_flag |= MSTATUS;
}
return(0);
}
/*
* Print the size of each message.
*/
int
messize(int *msgvec)
{
register struct message *mp;
register int *ip, mesg;
for (ip = msgvec; *ip != NULL; ip++) {
mesg = *ip;
mp = &message[mesg-1];
dot = mp;
printf("%d: %ld\n", mesg, mp->m_size);
}
return(0);
}
/*
* Quit quickly. If we are sourcing, just pop the input level
* by returning an error.
*/
int
rexit(int e)
{
if (sourcing)
return(1);
if (Tflag != NOSTR)
close(creat(Tflag, TEMPPERM));
if (!edit)
Verhogen();
exit(e ? e : rpterr);
/* NOTREACHED */
return (0); /* shut up lint and CC */
}
/*
* Set or display a variable value. Syntax is similar to that
* of csh.
*/
int
set(char **arglist)
{
register struct var *vp;
register char *cp, *cp2;
char varbuf[BUFSIZ], **ap, **p;
int errs, h, s;
if (argcount(arglist) == 0) {
for (h = 0, s = 1; h < HSHSIZE; h++)
for (vp = variables[h]; vp != NOVAR; vp = vp->v_link)
s++;
ap = (char **) salloc(s * sizeof *ap);
for (h = 0, p = ap; h < HSHSIZE; h++)
for (vp = variables[h]; vp != NOVAR; vp = vp->v_link)
*p++ = vp->v_name;
*p = NOSTR;
sort(ap);
for (p = ap; *p != NOSTR; p++)
if (((cp = value(*p)) != 0) && *cp)
printf("%s=\"%s\"\n", *p, cp);
else
printf("%s\n", *p);
return(0);
}
errs = 0;
for (ap = arglist; *ap != NOSTR; ap++) {
cp = *ap;
cp2 = varbuf;
while (*cp != '=' && *cp != '\0')
*cp2++ = *cp++;
*cp2 = '\0';
if (*cp == '\0')
cp = "";
else
cp++;
if (equal(varbuf, "")) {
printf(gettext("Non-null variable name required\n"));
errs++;
continue;
}
assign(varbuf, cp);
}
return(errs);
}
/*
* Unset a bunch of variable values.
*/
int
unset(char **arglist)
{
register int errs;
register char **ap;
errs = 0;
for (ap = arglist; *ap != NOSTR; ap++)
errs += deassign(*ap);
return(errs);
}
/*
* Add users to a group.
*/
int
group(char **argv)
{
register struct grouphead *gh;
register struct mgroup *gp;
register int h;
int s;
char **ap, *gname, **p;
if (argcount(argv) == 0) {
for (h = 0, s = 1; h < HSHSIZE; h++)
for (gh = groups[h]; gh != NOGRP; gh = gh->g_link)
s++;
ap = (char **) salloc(s * sizeof *ap);
for (h = 0, p = ap; h < HSHSIZE; h++)
for (gh = groups[h]; gh != NOGRP; gh = gh->g_link)
*p++ = gh->g_name;
*p = NOSTR;
sort(ap);
for (p = ap; *p != NOSTR; p++)
printgroup(*p);
return(0);
}
if (argcount(argv) == 1) {
printgroup(*argv);
return(0);
}
gname = *argv;
h = hash(gname);
if ((gh = findgroup(gname)) == NOGRP) {
if ((gh = (struct grouphead *)
calloc(sizeof (*gh), 1)) == NULL) {
panic("Failed to allocate memory for group");
}
gh->g_name = vcopy(gname);
gh->g_list = NOGE;
gh->g_link = groups[h];
groups[h] = gh;
}
/*
* Insert names from the command list into the group.
* Who cares if there are duplicates? They get tossed
* later anyway.
*/
for (ap = argv+1; *ap != NOSTR; ap++) {
if ((gp = (struct mgroup *)
calloc(sizeof (*gp), 1)) == NULL) {
panic("Failed to allocate memory for group");
}
gp->ge_name = vcopy(*ap);
gp->ge_link = gh->g_list;
gh->g_list = gp;
}
return(0);
}
/*
* Remove users from a group.
*/
int
ungroup(char **argv)
{
register struct grouphead *gh, **ghp;
register struct mgroup *gp, *gpnext;
register int h;
char **ap, *gname;
if (argcount(argv) == 0) {
printf("Must specify alias or group to remove\n");
return(1);
}
/*
* Remove names on the command list from the group list.
*/
for (ap = argv; *ap != NOSTR; ap++) {
gname = *ap;
h = hash(gname);
for (ghp = &groups[h]; *ghp != NOGRP; ghp = &((*ghp)->g_link)) {
gh = *ghp;
if (equal(gh->g_name, gname)) {
/* remove from list */
*ghp = gh->g_link;
/* free each member of gorup */
for (gp = gh->g_list; gp != NOGE; gp = gpnext) {
gpnext = gp->ge_link;
vfree(gp->ge_name);
free(gp);
}
vfree(gh->g_name);
free(gh);
break;
}
}
}
return(0);
}
/*
* Sort the passed string vecotor into ascending dictionary
* order.
*/
static void
sort(char **list)
{
register char **ap;
for (ap = list; *ap != NOSTR; ap++)
;
if (ap-list < 2)
return;
qsort((char *) list, (unsigned) (ap-list), sizeof *list, diction);
}
/*
* Do a dictionary order comparison of the arguments from
* qsort.
*/
static int
diction(const void *a, const void *b)
{
return(strcmp(*(char **)a, *(char **)b));
}
/*
* The do nothing command for comments.
*/
int
#ifdef __cplusplus
null(char *)
#else
/* ARGSUSED */
null(char *s)
#endif
{
return(0);
}
/*
* Print out the current edit file, if we are editing.
* Otherwise, print the name of the person who's mail
* we are reading.
*/
int
file(char **argv)
{
register char *cp;
int editing, mdot;
if (argv[0] == NOSTR) {
mdot = newfileinfo(1);
dot = &message[mdot - 1];
return(0);
}
/*
* Acker's! Must switch to the new file.
* We use a funny interpretation --
* # -- gets the previous file
* % -- gets the invoker's post office box
* %user -- gets someone else's post office box
* & -- gets invoker's mbox file
* string -- reads the given file
*/
cp = getfilename(argv[0], &editing);
if (cp == NOSTR)
return(-1);
if (setfile(cp, editing)) {
nstrcpy(origname, PATHSIZE, origprevfile);
return(-1);
}
mdot = newfileinfo(1);
dot = &message[mdot - 1];
return(0);
}
/*
* Evaluate the string given as a new mailbox name.
* Ultimately, we want this to support a number of meta characters.
* Possibly:
* % -- for my system mail box
* %user -- for user's system mail box
* # -- for previous file
* & -- get's invoker's mbox file
* file name -- for any other file
*/
static char *
getfilename(char *name, int *aedit)
{
register char *cp;
char savename[BUFSIZ];
char oldmailname[BUFSIZ];
char tmp[BUFSIZ];
/*
* Assume we will be in "edit file" mode, until
* proven wrong.
*/
*aedit = 1;
switch (*name) {
case '%':
*aedit = 0;
nstrcpy(prevfile, sizeof (prevfile), editfile);
nstrcpy(origprevfile, sizeof (origprevfile), origname);
if (name[1] != 0) {
nstrcpy(oldmailname, sizeof (oldmailname), mailname);
findmail(name+1);
cp = savestr(mailname);
nstrcpy(origname, PATHSIZE, cp);
nstrcpy(mailname, PATHSIZE, oldmailname);
return(cp);
}
nstrcpy(oldmailname, sizeof (oldmailname), mailname);
findmail(NULL);
cp = savestr(mailname);
nstrcpy(mailname, PATHSIZE, oldmailname);
nstrcpy(origname, PATHSIZE, cp);
return(cp);
case '#':
if (name[1] != 0)
goto regular;
if (prevfile[0] == 0) {
printf(gettext("No previous file\n"));
return(NOSTR);
}
cp = savestr(prevfile);
nstrcpy(prevfile, sizeof (prevfile), editfile);
nstrcpy(tmp, sizeof (tmp), origname);
nstrcpy(origname, PATHSIZE, origprevfile);
nstrcpy(origprevfile, sizeof (origprevfile), tmp);
return(cp);
case '&':
nstrcpy(prevfile, sizeof (prevfile), editfile);
nstrcpy(origprevfile, sizeof (origprevfile), origname);
if (name[1] == 0) {
cp=Getf("MBOX");
nstrcpy(origname, PATHSIZE, cp);
return(cp);
}
/* Fall into . . . */
default:
regular:
nstrcpy(prevfile, sizeof (prevfile), editfile);
nstrcpy(origprevfile, sizeof (origprevfile), origname);
cp = safeexpand(name);
nstrcpy(origname, PATHSIZE, cp);
if (cp[0] != '/') {
name = getcwd(NOSTR, PATHSIZE);
nstrcat(name, PATHSIZE, "/");
nstrcat(name, PATHSIZE, cp);
cp = name;
}
return(cp);
}
}
/*
* Expand file names like echo
*/
int
echo(register char **argv)
{
register char *cp;
int neednl = 0;
while (*argv != NOSTR) {
cp = *argv++;
if ((cp = expand(cp)) != NOSTR) {
neednl++;
printf("%s", cp);
if (*argv!=NOSTR)
putchar(' ');
}
}
if (neednl)
putchar('\n');
return(0);
}
/*
* Reply to a series of messages by simply mailing to the senders
* and not messing around with the To: and Cc: lists as in normal
* reply.
*/
int
Respond(int *msgvec)
{
if (reply2sender())
return(Resp1(msgvec, 0));
else
return(resp1(msgvec, 0));
}
int
Followup(int *msgvec)
{
if (reply2sender())
return(Resp1(msgvec, 1));
else
return(resp1(msgvec, 1));
}
int
replysender(int *msgvec)
{
return(Resp1(msgvec, 0));
}
static int
Resp1(int *msgvec, int useauthor)
{
struct header head;
struct message *mp;
register int s, *ap;
register char *cp, *cp2, *subject;
for (s = 0, ap = msgvec; *ap != 0; ap++) {
mp = &message[*ap - 1];
dot = mp;
cp = replyto(mp, NOSTRPTR);
s += strlen(cp) + 1;
}
if (s == 0)
return(0);
cp = (char *)salloc(s + 2);
head.h_to = cp;
for (ap = msgvec; *ap != 0; ap++) {
mp = &message[*ap - 1];
cp2 = replyto(mp, NOSTRPTR);
cp = copy(cp2, cp);
*cp++ = ' ';
}
*--cp = 0;
mp = &message[msgvec[0] - 1];
subject = hfield("subject", mp, addone);
head.h_seq = 1;
if (subject == NOSTR)
subject = hfield("subj", mp, addone);
head.h_subject = reedit(subject);
if (subject != NOSTR)
head.h_seq++;
head.h_cc = NOSTR;
head.h_bcc = NOSTR;
head.h_defopt = NOSTR;
head.h_others = NOSTRPTR;
mail1(&head, useauthor, NOSTR);
return(0);
}
/*
* Conditional commands. These allow one to parameterize one's
* .mailrc and do some things if sending, others if receiving.
*/
int
ifcmd(char **argv)
{
register char *cp;
if (cond != CANY) {
printf(gettext("Illegal nested \"if\"\n"));
return(1);
}
cond = CANY;
cp = argv[0];
switch (*cp) {
case 'r': case 'R':
cond = CRCV;
break;
case 's': case 'S':
cond = CSEND;
break;
case 't': case 'T':
cond = CTTY;
break;
default:
printf(gettext("Unrecognized if-keyword: \"%s\"\n"), cp);
return(1);
}
return(0);
}
/*
* Implement 'else'. This is pretty simple -- we just
* flip over the conditional flag.
*/
int
elsecmd(void)
{
switch (cond) {
case CANY:
printf(gettext("\"Else\" without matching \"if\"\n"));
return(1);
case CSEND:
cond = CRCV;
break;
case CRCV:
cond = CSEND;
break;
case CTTY:
cond = CNOTTY;
break;
case CNOTTY:
cond = CTTY;
break;
default:
printf(gettext("invalid condition encountered\n"));
cond = CANY;
break;
}
return(0);
}
/*
* End of if statement. Just set cond back to anything.
*/
int
endifcmd(void)
{
if (cond == CANY) {
printf(gettext("\"Endif\" without matching \"if\"\n"));
return(1);
}
cond = CANY;
return(0);
}
/*
* Set the list of alternate names.
*/
int
alternates(char **namelist)
{
register int c;
register char **ap, **ap2, *cp;
c = argcount(namelist) + 1;
if (c == 1) {
if (altnames == 0)
return(0);
for (ap = altnames; *ap; ap++)
printf("%s ", *ap);
printf("\n");
return (0);
}
if (altnames != 0)
free((char *)altnames);
if ((altnames = (char **)
calloc((unsigned)c, sizeof (char *))) == NULL)
panic("Failed to allocate memory");
for (ap = namelist, ap2 = altnames; *ap; ap++, ap2++) {
if ((cp = (char *)
calloc((unsigned)strlen(*ap) + 1, sizeof (char))) == NULL)
panic("Failed to allocate memory");
strcpy(cp, *ap);
*ap2 = cp;
}
*ap2 = 0;
return(0);
}
/*
* Figure out who to reply to.
* Return the real sender in *f.
*/
static char *
replyto(struct message *mp, char **f)
{
char *r, *rf;
if ((rf = skin(hfield("from", mp, addto)))==NOSTR)
rf = skin(addto(NOSTR, nameof(mp)));
if ((r = skin(hfield("reply-to", mp, addto)))==NOSTR)
r = rf;
if (f)
*f = rf;
return (r);
}
/*
* reply2sender - determine whether a "reply" command should reply to the
* sender of the messages, or to all the recipients of the
* message.
*
* With the advent of POSIX.2 compliance, this has become
* a bit more complicated, and so should be done in one
* place, for all to use.
*/
static int
reply2sender (void)
{
register int rep = (value("replyall") != NOSTR);
register int flp = (value("flipr") != NOSTR);
return((rep && !flp)|| (!rep && flp));
}