#include "readwrite.h"
#include "prioq.h"
#include "env.h"
#include "stralloc.h"
#include "subfd.h"
#include "substdio.h"
#include "getln.h"
#include "error.h"
#include "open.h"
#include "lock.h"
#include "gfrom.h"
#include "str.h"
#include "exit.h"
#include "myctime.h"
#include "maildir.h"

char *mbox;
char *mboxtmp;

stralloc filenames = {0};
prioq pq = {0};
prioq pq2 = {0};

stralloc line = {0};

stralloc ufline = {0};

char inbuf[SUBSTDIO_INSIZE];
char outbuf[SUBSTDIO_OUTSIZE];

#define FATAL "maildir2mbox: fatal: "
#define WARNING "maildir2mbox: warning: "

void die_nomem() { strerr_die2x(111,FATAL,"out of memory"); }

void main()
{
 substdio ssin;
 substdio ssout;
 struct prioq_elt pe;
 int fdoldmbox;
 int fdnewmbox;
 int fd;
 int match;
 int fdlock;

 umask(077);

 mbox = env_get("MAIL");
 if (!mbox) strerr_die2x(111,FATAL,"MAIL not set");
 mboxtmp = env_get("MAILTMP");
 if (!mboxtmp) strerr_die2x(111,FATAL,"MAILTMP not set");

 if (maildir_chdir() == -1)
   strerr_die1(111,FATAL,&maildir_chdir_err);
 maildir_clean(&filenames);
 if (maildir_scan(&pq,&filenames,1,1) == -1)
   strerr_die1(111,FATAL,&maildir_scan_err);

 if (!prioq_min(&pq,&pe)) _exit(0); /* nothing new */

 fdlock = open_append(mbox);
 if (fdlock == -1)
   strerr_die4sys(111,FATAL,"unable to lock ",mbox,": ");
 if (lock_ex(fdlock) == -1)
   strerr_die4sys(111,FATAL,"unable to lock ",mbox,": ");

 fdoldmbox = open_read(mbox);
 if (fdoldmbox == -1)
   strerr_die4sys(111,FATAL,"unable to read ",mbox,": ");

 fdnewmbox = open_trunc(mboxtmp);
 if (fdnewmbox == -1)
   strerr_die4sys(111,FATAL,"unable to create ",mboxtmp,": ");

 substdio_fdbuf(&ssin,read,fdoldmbox,inbuf,sizeof(inbuf));
 substdio_fdbuf(&ssout,write,fdnewmbox,outbuf,sizeof(outbuf));

 switch(substdio_copy(&ssout,&ssin))
  {
   case -2: strerr_die4sys(111,FATAL,"unable to read ",mbox,": ");
   case -3: strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
  }

 while (prioq_min(&pq,&pe))
  {
   prioq_delmin(&pq);
   if (!prioq_insert(&pq2,&pe)) die_nomem();

   fd = open_read(filenames.s + pe.id);
   if (fd == -1)
     strerr_die4sys(111,FATAL,"unable to read $MAILDIR/",filenames.s + pe.id,": ");
   substdio_fdbuf(&ssin,read,fd,inbuf,sizeof(inbuf));

   if (getln(&ssin,&line,&match,'\n') != 0)
     strerr_die4sys(111,FATAL,"unable to read $MAILDIR/",filenames.s + pe.id,": ");
   
   if (!stralloc_copys(&ufline,"From XXX ")) die_nomem();
   if (match)
     if (stralloc_starts(&line,"Return-Path: <"))
      {
       if (line.s[14] == '>')
	{
         if (!stralloc_copys(&ufline,"From MAILER-DAEMON ")) die_nomem();
	}
       else
	{
	 int i;
         if (!stralloc_ready(&ufline,line.len)) die_nomem();
         if (!stralloc_copys(&ufline,"From ")) die_nomem();
	 for (i = 14;i < line.len - 2;++i)
	   if ((line.s[i] == ' ') || (line.s[i] == '\t'))
	     ufline.s[ufline.len++] = '-';
	   else
	     ufline.s[ufline.len++] = line.s[i];
         if (!stralloc_cats(&ufline," ")) die_nomem();
	}
      }
   if (!stralloc_cats(&ufline,myctime(pe.dt))) die_nomem();
   if (substdio_put(&ssout,ufline.s,ufline.len) == -1)
     strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");

   while (match && line.len)
    {
     if (gfrom(line.s,line.len))
       if (substdio_puts(&ssout,">") == -1)
         strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
     if (substdio_put(&ssout,line.s,line.len) == -1)
       strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
     if (!match)
      {
       if (substdio_puts(&ssout,"\n") == -1)
         strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
       break;
      }
     if (getln(&ssin,&line,&match,'\n') != 0)
       strerr_die4sys(111,FATAL,"unable to read $MAILDIR/",filenames.s + pe.id,": ");
    }
   if (substdio_puts(&ssout,"\n"))
     strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");

   close(fd);
  }

 if (substdio_flush(&ssout) == -1)
   strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
 if (fsync(fdnewmbox) == -1)
   strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
 if (close(fdnewmbox) == -1) /* NFS dorks */
   strerr_die4sys(111,FATAL,"unable to write to ",mboxtmp,": ");
 if (rename(mboxtmp,mbox) == -1)
   strerr_die6(111,FATAL,"unable to move ",mboxtmp," to ",mbox,": ",&strerr_sys);
 
 while (prioq_min(&pq2,&pe))
  {
   prioq_delmin(&pq2);
   if (unlink(filenames.s + pe.id) == -1)
     strerr_warn4(WARNING,"$MAILDIR/",filenames.s + pe.id," will be delivered twice; unable to unlink: ",&strerr_sys);
  }

 _exit(0);
}


syntax highlighted by Code2HTML, v. 0.9.1