* Updated the wording of the output of a DOWN command ued to restart the server.
[citadel.git] / citadel / citmail.c
bloba665e8cc70abc390f8b2f050d48045fb18aaa88d
1 /*
2 * $Id$
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.
8 */
10 #include "sysdep.h"
11 #include <stdlib.h>
12 #include <unistd.h>
13 #include <ctype.h>
14 #include <stdio.h>
15 #include <signal.h>
16 #include <sys/types.h>
17 #include <sys/socket.h>
18 #include <sys/un.h>
19 #include <netdb.h>
20 #include <string.h>
21 #include <pwd.h>
22 #include <errno.h>
23 #include <stdarg.h>
24 #include <limits.h>
25 #include <libcitadel.h>
26 #include "citadel.h"
27 #ifndef HAVE_SNPRINTF
28 #include "snprintf.h"
29 #endif
30 #include "citadel_dirs.h"
32 int serv_sock;
33 int debug = 0;
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)
44 exit(signum);
48 int uds_connectsock(char *sockpath)
50 int s;
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);
58 if (s < 0) {
59 fprintf(stderr, "citmail: Can't create socket: %s\n",
60 strerror(errno));
61 exit(3);
64 if (connect(s, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
65 fprintf(stderr, "citmail: can't connect: %s\n",
66 strerror(errno));
67 close(s);
68 exit(3);
71 return s;
76 * input binary data from socket
78 void serv_read(char *buf, int bytes)
80 int len, rlen;
82 len = 0;
83 while (len < bytes) {
84 rlen = read(serv_sock, &buf[len], bytes - len);
85 if (rlen < 1) {
86 return;
88 len = len + rlen;
94 * send binary to server
96 void serv_write(char *buf, int nbytes)
98 int bytes_written = 0;
99 int retval;
100 while (bytes_written < nbytes) {
101 retval = write(serv_sock, &buf[bytes_written],
102 nbytes - bytes_written);
103 if (retval < 1) {
104 return;
106 bytes_written = bytes_written + retval;
113 * input string from socket - implemented in terms of serv_read()
115 void serv_gets(char *buf)
117 int i;
119 /* Read one character at a time.
121 for (i = 0;; i++) {
122 serv_read(&buf[i], 1);
123 if (buf[i] == '\n' || i == (SIZ-1))
124 break;
127 /* If we got a long line, discard characters until the newline.
129 if (i == (SIZ-1))
130 while (buf[i] != '\n')
131 serv_read(&buf[i], 1);
133 /* Strip all trailing nonprintables (crlf)
135 buf[i] = 0;
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));
148 serv_write("\n", 1);
153 void cleanup(int exitcode) {
154 char buf[1024];
156 if (exitcode != 0) {
157 fprintf(stderr, "citmail: error #%d occurred while sending mail.\n", exitcode);
158 fprintf(stderr, "Please check your Citadel configuration.\n");
160 serv_puts("QUIT");
161 serv_gets(buf);
162 exit(exitcode);
167 int main(int argc, char **argv) {
168 char buf[1024];
169 char fromline[1024];
170 FILE *fp;
171 int i;
172 struct passwd *pw;
173 int from_header = 0;
174 int in_body = 0;
175 int relh=0;
176 int home=0;
177 char relhome[PATH_MAX]="";
178 char ctdldir[PATH_MAX]=CTDLDIR;
179 char *sp, *ep;
180 char hostname[256];
181 char **recipients = NULL;
182 int num_recipients = 0;
183 int to_or_cc = 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")) {
189 debug = 1;
191 else if (!strcmp(argv[i], "-t")) {
192 read_recipients_from_headers = 1;
194 else if (argv[i][0] != '-') {
195 ++num_recipients;
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());
206 fp = tmpfile();
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 */
209 serv_gets(buf);
210 if (buf[0] != '2') {
211 fprintf(stderr, "%s\n", &buf[4]);
212 if (debug) fprintf(stderr, "citmail: could not connect to LMTP socket.\n");
213 cleanup(1);
216 sp = strchr (buf, ' ');
217 if (sp == NULL) {
218 if (debug) fprintf(stderr, "citmail: ould not calculate hostname.\n");
219 cleanup(2);
221 sp ++;
222 ep = strchr (sp, ' ');
223 if (ep == NULL) cleanup(3);
224 *ep = '\0';
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) ) {
230 in_body = 1;
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);
237 from_header = 1;
240 if (read_recipients_from_headers) {
241 add_these_recipients = NULL;
242 if ((isspace(buf[0])) && (to_or_cc)) {
243 add_these_recipients = buf;
245 else {
246 if ((!strncasecmp(buf, "To:", 3)) || (!strncasecmp(buf, "Cc:", 3))) {
247 to_or_cc = 1;
249 else {
250 to_or_cc = 0;
252 if (to_or_cc) {
253 add_these_recipients = &buf[3];
257 if (add_these_recipients) {
258 int num_recp_on_this_line;
259 char this_recp[256];
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);
265 striplt(this_recp);
266 if (!IsEmptyStr(this_recp)) {
267 ++num_recipients;
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);
281 serv_puts(buf);
282 do {
283 serv_gets(buf);
284 strcat(buf, " ");
285 } while (buf[3] == '-');
286 if (buf[0] != '2') cleanup(4);
288 snprintf(buf, sizeof buf, "MAIL %s", fromline);
289 serv_puts(buf);
290 serv_gets(buf);
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]);
295 serv_puts(buf);
296 serv_gets(buf);
297 free(recipients[i]);
299 free(recipients);
301 serv_puts("DATA");
302 serv_gets(buf);
303 if (buf[0]!='3') cleanup(6);
305 rewind(fp);
306 while (fgets(buf, sizeof buf, fp) != NULL) {
307 strip_trailing_nonprint(buf);
308 serv_puts(buf);
310 serv_puts(".");
311 serv_gets(buf);
312 if (buf[0] != '2') {
313 fprintf(stderr, "%s\n", &buf[4]);
314 cleanup(7);
316 else {
317 cleanup(0);
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.
324 return(0);