qmail-popup.c
changeset 0 068428edee47
equal deleted inserted replaced
-1:000000000000 0:068428edee47
       
     1 #include "commands.h"
       
     2 #include "fd.h"
       
     3 #include "sig.h"
       
     4 #include "stralloc.h"
       
     5 #include "substdio.h"
       
     6 #include "alloc.h"
       
     7 #include "wait.h"
       
     8 #include "str.h"
       
     9 #include "byte.h"
       
    10 #include "now.h"
       
    11 #include "fmt.h"
       
    12 #include "exit.h"
       
    13 #include "readwrite.h"
       
    14 #include "timeoutread.h"
       
    15 #include "timeoutwrite.h"
       
    16 
       
    17 void die() { _exit(1); }
       
    18 
       
    19 int saferead(fd,buf,len) int fd; char *buf; int len;
       
    20 {
       
    21   int r;
       
    22   r = timeoutread(1200,fd,buf,len);
       
    23   if (r <= 0) die();
       
    24   return r;
       
    25 }
       
    26 
       
    27 int safewrite(fd,buf,len) int fd; char *buf; int len;
       
    28 {
       
    29   int r;
       
    30   r = timeoutwrite(1200,fd,buf,len);
       
    31   if (r <= 0) die();
       
    32   return r;
       
    33 }
       
    34 
       
    35 char ssoutbuf[128];
       
    36 substdio ssout = SUBSTDIO_FDBUF(safewrite,1,ssoutbuf,sizeof ssoutbuf);
       
    37 
       
    38 char ssinbuf[128];
       
    39 substdio ssin = SUBSTDIO_FDBUF(saferead,0,ssinbuf,sizeof ssinbuf);
       
    40 
       
    41 void puts(s) char *s;
       
    42 {
       
    43   substdio_puts(&ssout,s);
       
    44 }
       
    45 void flush()
       
    46 {
       
    47   substdio_flush(&ssout);
       
    48 }
       
    49 void err(s) char *s;
       
    50 {
       
    51   puts("-ERR ");
       
    52   puts(s);
       
    53   puts("\r\n");
       
    54   flush();
       
    55 }
       
    56 
       
    57 void die_usage() { err("usage: popup hostname subprogram"); die(); }
       
    58 void die_nomem() { err("out of memory"); die(); }
       
    59 void die_pipe() { err("unable to open pipe"); die(); }
       
    60 void die_write() { err("unable to write pipe"); die(); }
       
    61 void die_fork() { err("unable to fork"); die(); }
       
    62 void die_childcrashed() { err("aack, child crashed"); }
       
    63 void die_badauth() { err("authorization failed"); }
       
    64 
       
    65 void err_syntax() { err("syntax error"); }
       
    66 void err_wantuser() { err("USER first"); }
       
    67 void err_authoriz() { err("authorization first"); }
       
    68 
       
    69 void okay() { puts("+OK \r\n"); flush(); }
       
    70 void pop3_quit() { okay(); die(); }
       
    71 
       
    72 
       
    73 char unique[FMT_ULONG + FMT_ULONG + 3];
       
    74 char *hostname;
       
    75 stralloc username = {0};
       
    76 int seenuser = 0;
       
    77 char **childargs;
       
    78 substdio ssup;
       
    79 char upbuf[128];
       
    80 
       
    81 
       
    82 void doanddie(user,userlen,pass)
       
    83 char *user;
       
    84 unsigned int userlen; /* including 0 byte */
       
    85 char *pass;
       
    86 {
       
    87   int child;
       
    88   int wstat;
       
    89   int pi[2];
       
    90  
       
    91   if (fd_copy(2,1) == -1) die_pipe();
       
    92   close(3);
       
    93   if (pipe(pi) == -1) die_pipe();
       
    94   if (pi[0] != 3) die_pipe();
       
    95   switch(child = fork()) {
       
    96     case -1:
       
    97       die_fork();
       
    98     case 0:
       
    99       close(pi[1]);
       
   100       sig_pipedefault();
       
   101       execvp(*childargs,childargs);
       
   102       _exit(1);
       
   103   }
       
   104   close(pi[0]);
       
   105   substdio_fdbuf(&ssup,write,pi[1],upbuf,sizeof upbuf);
       
   106   if (substdio_put(&ssup,user,userlen) == -1) die_write();
       
   107   if (substdio_put(&ssup,pass,str_len(pass) + 1) == -1) die_write();
       
   108   if (substdio_puts(&ssup,"<") == -1) die_write();
       
   109   if (substdio_puts(&ssup,unique) == -1) die_write();
       
   110   if (substdio_puts(&ssup,hostname) == -1) die_write();
       
   111   if (substdio_put(&ssup,">",2) == -1) die_write();
       
   112   if (substdio_flush(&ssup) == -1) die_write();
       
   113   close(pi[1]);
       
   114   byte_zero(pass,str_len(pass));
       
   115   byte_zero(upbuf,sizeof upbuf);
       
   116   if (wait_pid(&wstat,child) == -1) die();
       
   117   if (wait_crashed(wstat)) die_childcrashed();
       
   118   if (wait_exitcode(wstat)) die_badauth();
       
   119   die();
       
   120 }
       
   121 void pop3_greet()
       
   122 {
       
   123   char *s;
       
   124   s = unique;
       
   125   s += fmt_uint(s,getpid());
       
   126   *s++ = '.';
       
   127   s += fmt_ulong(s,(unsigned long) now());
       
   128   *s++ = '@';
       
   129   *s++ = 0;
       
   130   puts("+OK <");
       
   131   puts(unique);
       
   132   puts(hostname);
       
   133   puts(">\r\n");
       
   134   flush();
       
   135 }
       
   136 void pop3_user(arg) char *arg;
       
   137 {
       
   138   if (!*arg) { err_syntax(); return; }
       
   139   okay();
       
   140   seenuser = 1;
       
   141   if (!stralloc_copys(&username,arg)) die_nomem(); 
       
   142   if (!stralloc_0(&username)) die_nomem(); 
       
   143 }
       
   144 void pop3_pass(arg) char *arg;
       
   145 {
       
   146   if (!seenuser) { err_wantuser(); return; }
       
   147   if (!*arg) { err_syntax(); return; }
       
   148   doanddie(username.s,username.len,arg);
       
   149 }
       
   150 void pop3_apop(arg) char *arg;
       
   151 {
       
   152   char *space;
       
   153   space = arg + str_chr(arg,' ');
       
   154   if (!*space) { err_syntax(); return; }
       
   155   *space++ = 0;
       
   156   doanddie(arg,space - arg,space);
       
   157 }
       
   158 
       
   159 struct commands pop3commands[] = {
       
   160   { "user", pop3_user, 0 }
       
   161 , { "pass", pop3_pass, 0 }
       
   162 , { "apop", pop3_apop, 0 }
       
   163 , { "quit", pop3_quit, 0 }
       
   164 , { "noop", okay, 0 }
       
   165 , { 0, err_authoriz, 0 }
       
   166 } ;
       
   167 
       
   168 void main(argc,argv)
       
   169 int argc;
       
   170 char **argv;
       
   171 {
       
   172   sig_alarmcatch(die);
       
   173   sig_pipeignore();
       
   174  
       
   175   hostname = argv[1];
       
   176   if (!hostname) die_usage();
       
   177   childargs = argv + 2;
       
   178   if (!*childargs) die_usage();
       
   179  
       
   180   pop3_greet();
       
   181   commands(&ssin,pop3commands);
       
   182   die();
       
   183 }