implement the event loop, add a stopper.
[apps/pfixtools.git] / postfix.c
1 /******************************************************************************/
2 /*          postlicyd: a postfix policy daemon with a lot of features         */
3 /*          ~~~~~~~~~                                                         */
4 /*  ________________________________________________________________________  */
5 /*                                                                            */
6 /*  Redistribution and use in source and binary forms, with or without        */
7 /*  modification, are permitted provided that the following conditions        */
8 /*  are met:                                                                  */
9 /*                                                                            */
10 /*  1. Redistributions of source code must retain the above copyright         */
11 /*     notice, this list of conditions and the following disclaimer.          */
12 /*  2. Redistributions in binary form must reproduce the above copyright      */
13 /*     notice, this list of conditions and the following disclaimer in the    */
14 /*     documentation and/or other materials provided with the distribution.   */
15 /*  3. The names of its contributors may not be used to endorse or promote    */
16 /*     products derived from this software without specific prior written     */
17 /*     permission.                                                            */
18 /*                                                                            */
19 /*  THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND   */
20 /*  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE     */
21 /*  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR        */
22 /*  PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS    */
23 /*  BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR    */
24 /*  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF      */
25 /*  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS  */
26 /*  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN   */
27 /*  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)   */
28 /*  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF    */
29 /*  THE POSSIBILITY OF SUCH DAMAGE.                                           */
30 /******************************************************************************/
31
32 /*
33  * Copyright © 2006-2007 Pierre Habouzit
34  */
35
36 #include <errno.h>
37 #include <stdbool.h>
38 #include <syslog.h>
39 #include <unistd.h>
40
41 #include "job.h"
42 #include "postfix.h"
43 #include "buffer.h"
44
45 struct jpriv_t {
46     buffer_t ibuf;
47     buffer_t obuf;
48 };
49
50 jpriv_t *postfix_jpriv_init(jpriv_t *jp)
51 {
52     buffer_init(&jp->ibuf);
53     buffer_init(&jp->obuf);
54     return jp;
55 }
56 void postfix_jpriv_wipe(jpriv_t *jp)
57 {
58     buffer_wipe(&jp->ibuf);
59     buffer_wipe(&jp->obuf);
60 }
61 DO_NEW(jpriv_t, postfix_jpriv);
62 DO_DELETE(jpriv_t, postfix_jpriv);
63
64
65 void postfix_start(job_t *listener)
66 {
67     job_t *job;
68
69     job = job_accept(listener, JOB_READ);
70     if (!job)
71         return;
72
73     job->jdata = postfix_jpriv_new();
74 }
75
76 void postfix_stop(job_t *job)
77 {
78     postfix_jpriv_delete(&job->jdata);
79 }
80
81 void postfix_process(job_t *job)
82 {
83     int nb;
84
85     switch (job->state) {
86       case JOB_LISTEN:
87         return postfix_start(job);
88
89       case JOB_WRITE:
90         nb = write(job->fd, job->jdata->obuf.data, job->jdata->obuf.len);
91         if (nb < 0) {
92             if ((job->error = errno != EINTR && errno != EAGAIN)) {
93                 syslog(LOG_ERR, "unexpected problem on the socket: %m");
94             }
95             return;
96         }
97
98         buffer_consume(&job->jdata->obuf, nb);
99         if (job->jdata->obuf.len)
100             return;
101
102         job_update_state(job, JOB_READ);
103
104         /* fall through */
105
106       case JOB_READ:
107         nb = buffer_read(&job->jdata->ibuf, job->fd, -1);
108         if (nb < 0) {
109             if ((job->error = errno != EINTR && errno != EAGAIN)) {
110                 syslog(LOG_ERR, "unexpected problem on the socket: %m");
111             }
112             return;
113         }
114         if (nb == 0) {
115             syslog(LOG_ERR, "unexpected eof");
116             job->error = true;
117             return;
118         }
119
120         if (!strstr(job->jdata->ibuf.data, "\r\n\r\n"))
121             return;
122
123         /* TODO: do the parse */
124         job_update_state(job, JOB_IDLE);
125         return;
126
127       default:
128         job->error = true;
129         return;
130     }
131 }