4 * This program attempts to act like a local MDA if you're using sendmail or
5 * some other non-Citadel MTA. It basically just contacts the Citadel LMTP
6 * listener on a unix domain socket and transmits the message.
16 #include <sys/types.h>
17 #include <sys/socket.h>
25 #include <libcitadel.h>
30 #include "citadel_dirs.h"
35 void strip_trailing_nonprint(char *buf
)
37 while ( (!IsEmptyStr(buf
)) && (!isprint(buf
[strlen(buf
) - 1])) )
38 buf
[strlen(buf
) - 1] = 0;
42 void timeout(int signum
)
48 int uds_connectsock(char *sockpath
)
51 struct sockaddr_un addr
;
53 memset(&addr
, 0, sizeof(addr
));
54 addr
.sun_family
= AF_UNIX
;
55 strncpy(addr
.sun_path
, sockpath
, sizeof addr
.sun_path
);
57 s
= socket(AF_UNIX
, SOCK_STREAM
, 0);
59 fprintf(stderr
, "citmail: Can't create socket: %s\n",
64 if (connect(s
, (struct sockaddr
*) &addr
, sizeof(addr
)) < 0) {
65 fprintf(stderr
, "citmail: can't connect: %s\n",
76 * input binary data from socket
78 void serv_read(char *buf
, int bytes
)
84 rlen
= read(serv_sock
, &buf
[len
], bytes
- len
);
94 * send binary to server
96 void serv_write(char *buf
, int nbytes
)
98 int bytes_written
= 0;
100 while (bytes_written
< nbytes
) {
101 retval
= write(serv_sock
, &buf
[bytes_written
],
102 nbytes
- bytes_written
);
106 bytes_written
= bytes_written
+ retval
;
113 * input string from socket - implemented in terms of serv_read()
115 void serv_gets(char *buf
)
119 /* Read one character at a time.
122 serv_read(&buf
[i
], 1);
123 if (buf
[i
] == '\n' || i
== (SIZ
-1))
127 /* If we got a long line, discard characters until the newline.
130 while (buf
[i
] != '\n')
131 serv_read(&buf
[i
], 1);
133 /* Strip all trailing nonprintables (crlf)
136 strip_trailing_nonprint(buf
);
137 if (debug
) fprintf(stderr
, "> %s\n", buf
);
142 * send line to server - implemented in terms of serv_write()
144 void serv_puts(char *buf
)
146 if (debug
) fprintf(stderr
, "< %s\n", buf
);
147 serv_write(buf
, strlen(buf
));
153 void cleanup(int exitcode
) {
157 fprintf(stderr
, "citmail: error #%d occurred while sending mail.\n", exitcode
);
158 fprintf(stderr
, "Please check your Citadel configuration.\n");
167 int main(int argc
, char **argv
) {
177 char relhome
[PATH_MAX
]="";
178 char ctdldir
[PATH_MAX
]=CTDLDIR
;
181 char **recipients
= NULL
;
182 int num_recipients
= 0;
184 int read_recipients_from_headers
= 0;
185 char *add_these_recipients
= NULL
;
187 for (i
=1; i
<argc
; ++i
) {
188 if (!strcmp(argv
[i
], "-d")) {
191 else if (!strcmp(argv
[i
], "-t")) {
192 read_recipients_from_headers
= 1;
194 else if (argv
[i
][0] != '-') {
196 recipients
= realloc(recipients
, (num_recipients
* sizeof (char *)));
197 recipients
[num_recipients
- 1] = strdup(argv
[i
]);
201 /* TODO: should we be able to calculate relative dirs? */
202 calc_dirs_n_files(relh
, home
, relhome
, ctdldir
, 0);
204 pw
= getpwuid(getuid());
207 if (fp
== NULL
) return(errno
);
208 serv_sock
= uds_connectsock(file_lmtp_socket
); /* FIXME: if called as 'sendmail' connect to file_lmtp_unfiltered_socket */
211 fprintf(stderr
, "%s\n", &buf
[4]);
212 if (debug
) fprintf(stderr
, "citmail: could not connect to LMTP socket.\n");
216 sp
= strchr (buf
, ' ');
218 if (debug
) fprintf(stderr
, "citmail: ould not calculate hostname.\n");
222 ep
= strchr (sp
, ' ');
223 if (ep
== NULL
) cleanup(3);
225 strncpy(hostname
, sp
, sizeof hostname
);
227 snprintf(fromline
, sizeof fromline
, "From: %s@%s", pw
->pw_name
, hostname
);
228 while (fgets(buf
, 1024, stdin
) != NULL
) {
229 if ( ( (buf
[0] == 13) || (buf
[0] == 10)) && (in_body
== 0) ) {
231 if (from_header
== 0) {
232 fprintf(fp
, "%s%s", fromline
, buf
);
235 if (in_body
== 0 && !strncasecmp(buf
, "From:", 5)) {
236 strcpy(fromline
, buf
);
240 if (read_recipients_from_headers
) {
241 add_these_recipients
= NULL
;
242 if ((isspace(buf
[0])) && (to_or_cc
)) {
243 add_these_recipients
= buf
;
246 if ((!strncasecmp(buf
, "To:", 3)) || (!strncasecmp(buf
, "Cc:", 3))) {
253 add_these_recipients
= &buf
[3];
257 if (add_these_recipients
) {
258 int num_recp_on_this_line
;
261 num_recp_on_this_line
= num_tokens(add_these_recipients
, ',');
262 for (i
=0; i
<num_recp_on_this_line
; ++i
) {
263 extract_token(this_recp
, add_these_recipients
,
264 i
, ',', sizeof this_recp
);
266 if (!IsEmptyStr(this_recp
)) {
268 recipients
= realloc(recipients
,
269 (num_recipients
* sizeof (char *)));
270 recipients
[num_recipients
- 1] = strdup(this_recp
);
276 fprintf(fp
, "%s", buf
);
278 strip_trailing_nonprint(fromline
);
280 sprintf(buf
, "LHLO %s", hostname
);
285 } while (buf
[3] == '-');
286 if (buf
[0] != '2') cleanup(4);
288 snprintf(buf
, sizeof buf
, "MAIL %s", fromline
);
291 if (buf
[0] != '2') cleanup(5);
293 for (i
=0; i
<num_recipients
; ++i
) {
294 snprintf(buf
, sizeof buf
, "RCPT To: %s", recipients
[i
]);
303 if (buf
[0]!='3') cleanup(6);
306 while (fgets(buf
, sizeof buf
, fp
) != NULL
) {
307 strip_trailing_nonprint(buf
);
313 fprintf(stderr
, "%s\n", &buf
[4]);
320 /* We won't actually reach this statement but the compiler will
321 * display a spurious warning about an invalid return type if
322 * we don't return an int.