b3c853d31f9d2b0b8c709ce302bb64f9c5853ba8
[citadel.git] / webcit / tcp_sockets.c
1 /*
2  * tcp_sockets.c
3  * 
4  * TCP socket module for WebCit
5  *
6  * $Id$
7  */
8
9
10 #include <ctype.h>
11 #include <stdlib.h>
12 #include <unistd.h>
13 #include <stdio.h>
14 #include <fcntl.h>
15 #include <signal.h>
16 #include <sys/types.h>
17 #include <sys/wait.h>
18 #include <sys/socket.h>
19 #include <sys/time.h>
20 #include <limits.h>
21 #include <netinet/in.h>
22 #include <arpa/inet.h>
23 #include <sys/un.h>
24 #include <netdb.h>
25 #include <string.h>
26 #include <pwd.h>
27 #include <errno.h>
28 #include <stdarg.h>
29 #include <pthread.h>
30 #include <signal.h>
31 #include "webcit.h"
32 #include "webserver.h"
33
34
35 #ifndef INADDR_NONE
36 #define INADDR_NONE 0xffffffff
37 #endif
38
39 RETSIGTYPE timeout(int signum)
40 {
41         lprintf(1, "Connection timed out.\n");
42         exit(3);
43 }
44
45
46
47 /*
48  * Connect a unix domain socket
49  */
50 int uds_connectsock(char *sockpath)
51 {
52         struct sockaddr_un addr;
53         int s;
54
55         memset(&addr, 0, sizeof(addr));
56         addr.sun_family = AF_UNIX;
57         strncpy(addr.sun_path, sockpath, sizeof addr.sun_path);
58
59         s = socket(AF_UNIX, SOCK_STREAM, 0);
60         if (s < 0) {
61                 lprintf(1, "Can't create socket: %s\n",
62                         strerror(errno));
63                 return(-1);
64         }
65
66         if (connect(s, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
67                 lprintf(1, "Can't connect: %s\n",
68                         strerror(errno));
69                 close(s);
70                 return(-1);
71         }
72
73         return s;
74 }
75
76
77 /*
78  * Connect a TCP/IP socket
79  */
80 int tcp_connectsock(char *host, char *service)
81 {
82         struct hostent *phe;
83         struct servent *pse;
84         struct protoent *ppe;
85         struct sockaddr_in sin;
86         int s;
87
88         memset(&sin, 0, sizeof(sin));
89         sin.sin_family = AF_INET;
90
91         pse = getservbyname(service, "tcp");
92         if (pse) {
93                 sin.sin_port = pse->s_port;
94         } else if ((sin.sin_port = htons((u_short) atoi(service))) == 0) {
95                 lprintf(1, "Can't get %s service entry\n", service);
96                 return (-1);
97         }
98         phe = gethostbyname(host);
99         if (phe) {
100                 memcpy(&sin.sin_addr, phe->h_addr, phe->h_length);
101         } else if ((sin.sin_addr.s_addr = inet_addr(host)) == INADDR_NONE) {
102                 lprintf(1, "Can't get %s host entry: %s\n",
103                         host, strerror(errno));
104                 return (-1);
105         }
106         if ((ppe = getprotobyname("tcp")) == 0) {
107                 lprintf(1, "Can't get TCP protocol entry: %s\n",
108                         strerror(errno));
109                 return (-1);
110         }
111
112         s = socket(PF_INET, SOCK_STREAM, ppe->p_proto);
113         if (s < 0) {
114                 lprintf(1, "Can't create socket: %s\n", strerror(errno));
115                 return (-1);
116         }
117         signal(SIGALRM, timeout);
118         alarm(30);
119
120         if (connect(s, (struct sockaddr *) &sin, sizeof(sin)) < 0) {
121                 lprintf(1, "Can't connect to %s.%s: %s\n",
122                         host, service, strerror(errno));
123                 close(s);
124                 return (-1);
125         }
126         alarm(0);
127         signal(SIGALRM, SIG_IGN);
128
129         return (s);
130 }
131
132
133
134
135 /*
136  * Input binary data from socket
137  */
138 void serv_read(char *buf, int bytes)
139 {
140         int len, rlen;
141
142         len = 0;
143         while (len < bytes) {
144                 rlen = read(WC->serv_sock, &buf[len], bytes - len);
145                 if (rlen < 1) {
146                         lprintf(1, "Server connection broken: %s\n",
147                                 strerror(errno));
148                         close(WC->serv_sock);
149                         WC->serv_sock = (-1);
150                         WC->connected = 0;
151                         WC->logged_in = 0;
152                         return;
153                 }
154                 len = len + rlen;
155         }
156 }
157
158
159 /*
160  * input string from pipe
161  */
162 void serv_gets(char *strbuf)
163 {
164         int ch, len;
165         char buf[2];
166
167         len = 0;
168         strcpy(strbuf, "");
169         do {
170                 serv_read(&buf[0], 1);
171                 ch = buf[0];
172                 strbuf[len++] = ch;
173         } while ((ch != 10) && (ch != 0) && (len < (SIZ-1)));
174         if (strbuf[len-1] == 10) strbuf[--len] = 0;
175         if (strbuf[len-1] == 13) strbuf[--len] = 0;
176         /* lprintf(9, ">%s\n", strbuf); */
177 }
178
179
180
181 /*
182  * send binary to server
183  */
184 void serv_write(char *buf, int nbytes)
185 {
186         int bytes_written = 0;
187         int retval;
188         while (bytes_written < nbytes) {
189                 retval = write(WC->serv_sock, &buf[bytes_written],
190                                nbytes - bytes_written);
191                 if (retval < 1) {
192                         lprintf(1, "Server connection broken: %s\n",
193                                 strerror(errno));
194                         close(WC->serv_sock);
195                         WC->serv_sock = (-1);
196                         WC->connected = 0;
197                         WC->logged_in = 0;
198                         return;
199                 }
200                 bytes_written = bytes_written + retval;
201         }
202 }
203
204
205 /*
206  * send line to server
207  */
208 void serv_puts(char *string)
209 {
210         char buf[SIZ];
211
212         sprintf(buf, "%s\n", string);
213         serv_write(buf, strlen(buf));
214 }
215
216
217 /*
218  * convenience function to send stuff to the server
219  */
220 void serv_printf(const char *format,...)
221 {
222         va_list arg_ptr;
223         char buf[SIZ];
224
225         va_start(arg_ptr, format);
226         vsprintf(buf, format, arg_ptr);
227         va_end(arg_ptr);
228
229         strcat(buf, "\n");
230         serv_write(buf, strlen(buf));
231         /* lprintf(9, "<%s", buf); */
232 }