/*
 * Copyright (c) 1989 Regents of the University of California.
 * All rights reserved.  The Berkeley software License Agreement
 * specifies the terms and conditions for redistribution.
 */

#include <popper.h>
RCSID("$Id: pop_send.c,v 1.18 1997/05/11 11:04:35 assar Exp $");

/*
 *  sendline:   Send a line of a multi-line response to a client.
 */
static
void
pop_sendline(POP *p, char *buffer)
{
    char        *   bp;

    /*  Byte stuff lines that begin with the termination octet */
    if (*buffer == POP_TERMINATE) 
      fputc(POP_TERMINATE,p->output);

    /*  Look for a <NL> in the buffer */
    if ((bp = strchr(buffer, '\n')))
      *bp = 0;

    /*  Send the line to the client */
    fputs(buffer,p->output);

#ifdef DEBUG
    if(p->debug)
      pop_log(p,POP_DEBUG,"Sending line \"%s\"",buffer);
#endif /* DEBUG */

    /*  Put a <CR><NL> if a newline was removed from the buffer */
    if (bp)
      fputs ("\r\n",p->output);
}

/* 
 *  send:   Send the header and a specified number of lines 
 *          from a mail message to a POP client.
 */

int
pop_send(POP *p)
{
    MsgInfoList         *   mp;         /*  Pointer to message info list */
    int		            msg_num;
    int			    msg_lines;
    char                    buffer[MAXMSGLINELEN];
#ifdef RETURN_PATH_HANDLING
    char		*   return_path_adr;
    char		*   return_path_end;
    int			    return_path_sent;
    int			    return_path_linlen;
#endif

    /*  Convert the first parameter into an integer */
    msg_num = atoi(p->pop_parm[1]);

    /*  Is requested message out of range? */
    if ((msg_num < 1) || (msg_num > p->msg_count))
        return (pop_msg (p,POP_FAILURE,"Message %d does not exist.",msg_num));

    /*  Get a pointer to the message in the message list */
    mp = &p->mlp[msg_num-1];

    /*  Is the message flagged for deletion? */
    if (mp->del_flag)
        return (pop_msg (p,POP_FAILURE,
            "Message %d has been deleted.",msg_num));

    /*  If this is a TOP command, get the number of lines to send */
    if (strcmp(p->pop_command,"top") == 0) {
        /*  Convert the second parameter into an integer */
        msg_lines = atoi(p->pop_parm[2]);
    }
    else {
        /*  Assume that a RETR (retrieve) command was issued */
        msg_lines = -1;
        /*  Flag the message as retreived */
        mp->retr_flag = TRUE;
    }
    
    /*  Display the number of bytes in the message */
    pop_msg(p, POP_SUCCESS, "%ld octets", mp->length);

    /*  Position to the start of the message */
    fseek(p->drop,mp->offset,0);

    /*  Skip the first line (the sendmail "From" line) */
    fgets (buffer,MAXMSGLINELEN,p->drop);

#ifdef RETURN_PATH_HANDLING
    return_path_sent = 0;
    if (strncmp(buffer,"From ",5) == 0) {
	return_path_linlen = strlen(buffer);
	for (return_path_adr = buffer+5;
	     (*return_path_adr == ' ' || *return_path_adr == '\t') &&
	     return_path_adr < buffer + return_path_linlen;
	     return_path_adr++)
	    ;
	if (return_path_adr < buffer + return_path_linlen) {
	    if ((return_path_end = strchr(return_path_adr, ' ')) != NULL)
		*return_path_end = '\0';
	    if (strlen(return_path_adr) != 0 && *return_path_adr != '\n') {
		static char tmpbuf[MAXMSGLINELEN + 20];
		strcpy(tmpbuf, "Return-Path: ");
		strcat(tmpbuf, return_path_adr);
		strcat(tmpbuf, "\n");
		if (strlen(tmpbuf) < MAXMSGLINELEN) {
		    pop_sendline (p,tmpbuf);
		    if (hangup)
			return (pop_msg (p,POP_FAILURE,"SIGHUP or SIGPIPE flagged"));
		    return_path_sent++;
		}
	    }
	}
    }
#endif

    /*  Send the header of the message followed by a blank line */
    while (fgets(buffer,MAXMSGLINELEN,p->drop)) {
#ifdef RETURN_PATH_HANDLING
	/* Don't send existing Return-Path-header if already sent own */
	if (strncasecmp(buffer,"Return-Path:", 12) != 0 ||
	    !return_path_sent)
#endif
        pop_sendline (p,buffer);
        /*  A single newline (blank line) signals the 
            end of the header.  sendline() converts this to a NULL, 
            so that's what we look for. */
        if (*buffer == 0) break;
        if (hangup)
          return (pop_msg (p,POP_FAILURE,"SIGHUP or SIGPIPE flagged"));
    }
    /*  Send the message body */
    {
	int blank_line = 0;
	while (fgets(buffer,MAXMSGLINELEN,p->drop)) {
	    /*  Look for the start of the next message */
	    if (blank_line && strncmp(buffer,"From ",5) == 0) break;
	    blank_line = (strncmp(buffer, "\n", 1) == 0);
	    /*  Decrement the lines sent (for a TOP command) */
	    if (msg_lines >= 0 && msg_lines-- == 0) break;
	    pop_sendline(p,buffer);
	    if (hangup)
		return (pop_msg (p,POP_FAILURE,"SIGHUP or SIGPIPE flagged"));
	}
    }
    /*  "." signals the end of a multi-line transmission */
    fputs(".\r\n",p->output);
    fflush(p->output);

    return(POP_SUCCESS);
}
