]> git.armaanb.net Git - sic.git/blob - sic.c
applied Matthias-Christian Ott's remark about trailing newlines
[sic.git] / sic.c
1 /* © 2005-2008 Anselm R Garbe <garbeam at gmail dot com>
2  * © 2005 Nico Golde <nico at ngolde dot de>
3  * See LICENSE file for license details. */
4 #include <errno.h>
5 #include <netdb.h>
6 #include <netinet/in.h>
7 #include <stdarg.h>
8 #include <stdio.h>
9 #include <stdlib.h>
10 #include <string.h>
11 #include <time.h>
12 #include <unistd.h>
13 #include <sys/socket.h>
14 #include <sys/time.h>
15
16 #define PINGTIMEOUT 300
17 #define MAXMSG      4096
18
19 static void die(const char *errstr, ...);
20 static void printl(char *channel, char *msg);
21 static void privmsg(char *channel, char *msg);
22 static void parsein(char *msg);
23 static void parsesrv(char *msg);
24 static int readl(int fd, unsigned int len, char *buf);
25
26 static char *host = "irc.oftc.net";
27 static unsigned short port = 6667;
28 static char *password = NULL;
29 static char nick[32];
30
31 static char bufin[MAXMSG], bufout[MAXMSG];
32 static char channel[256];
33 static int srv;
34 static time_t trespond;
35
36 void
37 die(const char *errstr, ...) {
38         va_list ap;
39
40         va_start(ap, errstr);
41         vfprintf(stderr, errstr, ap);
42         va_end(ap);
43         exit(EXIT_FAILURE);
44 }
45
46 void
47 printl(char *channel, char *msg) {
48         static char timestr[18];
49         time_t t = time(0);
50
51         strftime(timestr, sizeof timestr, "%D %R", localtime(&t));
52         fprintf(stdout, "%-12.12s: %s %s\n", channel, timestr, msg);
53 }
54
55 void
56 privmsg(char *channel, char *msg) {
57         if(channel[0] == 0)
58                 return;
59         snprintf(bufout, sizeof bufout, "<%s> %s", nick, msg);
60         printl(channel, bufout);
61         snprintf(bufout, sizeof bufout, "PRIVMSG %s :%s\r\n", channel, msg);
62         write(srv, bufout, strlen(bufout));
63 }
64
65 void
66 parsein(char *msg) {
67         char *p;
68
69         if(msg[0] == 0)
70                 return;
71         if(msg[0] != ':') {
72                 privmsg(channel, msg);
73                 return;
74         }
75         if(!strncmp(msg + 1, "j ", 2) && (msg[3] == '#'))
76                 snprintf(bufout, sizeof bufout, "JOIN %s\r\n", msg + 3);
77         else if(!strncmp(msg + 1, "l ", 2))
78                 snprintf(bufout, sizeof bufout, "PART %s :sic - 250 LOC are too much!\r\n", msg + 3);
79         else if(!strncmp(msg + 1, "m ", 2)) {
80                 if((p = strchr(msg + 3, ' ')))
81                         *(p++) = 0;
82                 privmsg(msg + 3, p);
83                 return;
84         }
85         else if(!strncmp(msg + 1, "s ", 2)) {
86                 strncpy(channel, msg + 3, sizeof channel);
87                 return;
88         }
89         else
90                 snprintf(bufout, sizeof bufout, "%s\r\n", msg + 1);
91         write(srv, bufout, strlen(bufout));
92 }
93
94 void
95 parsesrv(char *msg) {
96         char *chan, *cmd, *p, *txt, *usr; 
97
98         txt = NULL;
99         usr = host;
100         if(!msg || !(*msg))
101                 return;
102         if(msg[0] != ':')
103                 cmd = msg;
104         else {
105                 if(!(p = strchr(msg, ' ')))
106                         return;
107                 *p = 0;
108                 usr = msg + 1;
109                 cmd = ++p;
110                 if((p = strchr(usr, '!')))
111                         *p = 0;
112         }
113         for(p = cmd; *p; p++) /* remove CRLFs */
114                 if(*p == '\r' || *p == '\n')
115                         *p = 0;
116         if((p = strchr(cmd, ':'))) {
117                 *p = 0;
118                 txt = ++p;
119         }
120         if(!strncmp("PONG", cmd, 4))
121                 return;
122         if(!strncmp("PRIVMSG", cmd, 7) && txt) {
123                 if(!(p = strchr(cmd, ' ')))
124                         return;
125                 *p = 0;
126                 chan = ++p;
127                 for(; *p && *p != ' '; p++);
128                 *p = 0;
129                 snprintf(bufout, sizeof bufout, "<%s> %s", usr, txt);
130                 printl(chan, bufout);
131         }
132         else if(!strncmp("PING", cmd, 4) && txt) {
133                 snprintf(bufout, sizeof bufout, "PONG %s\r\n", txt);
134                 write(srv, bufout, strlen(bufout));
135         }
136         else {
137                 snprintf(bufout, sizeof bufout, ">< %s: %s", cmd, txt ? txt : "");
138                 printl(usr, bufout);
139                 if(!strncmp("NICK", cmd, 4) && !strncmp(usr, nick, sizeof nick) && txt)
140                         strncpy(nick, txt, sizeof nick);
141         }
142 }
143
144 int
145 readl(int fd, unsigned int len, char *buf) {
146         unsigned int i = 0;
147         char c;
148
149         do {
150                 if(read(fd, &c, sizeof(char)) != sizeof(char))
151                         return -1;
152                 buf[i++] = c;
153         }
154         while(c != '\n' && i < len);
155         buf[i - 1] = 0;
156         return 0;
157 }
158
159
160 int
161 main(int argc, char *argv[]) {
162         int i;
163         struct timeval tv;
164         struct hostent *hp;
165         static struct sockaddr_in addr;  /* initially filled with 0's */
166         char ping[256];
167         fd_set rd;
168
169         strncpy(nick, getenv("USER"), sizeof nick);
170         for(i = 1; i < argc; i++)
171                 if(!strncmp(argv[i], "-h", 3)) {
172                         if(++i < argc) host = argv[i];
173                 }
174                 else if(!strncmp(argv[i], "-p", 3)) {
175                         if(++i < argc) port = (unsigned short)atoi(argv[i]);
176                 }
177                 else if(!strncmp(argv[i], "-n", 3)) {
178                         if(++i < argc) strncpy(nick, argv[i], sizeof nick);
179                 }
180                 else if(!strncmp(argv[i], "-k", 3)) {
181                         if(++i < argc) password = argv[i];
182                 }
183                 else if(!strncmp(argv[i], "-v", 3))
184                         die("sic-"VERSION", © 2005-2008 Anselm R Garbe, Nico Golde\n");
185                 else
186                         die("usage: sic [-h host] [-p port] [-n nick] [-k keyword] [-v]\n");
187
188         /* init */
189         if((srv = socket(AF_INET, SOCK_STREAM, 0)) < 0)
190                 die("error: cannot connect host '%s'\n", host);
191         if(NULL == (hp = gethostbyname(host)))
192                 die("error: cannot resolve hostname '%s'\n", host);
193         addr.sin_family = AF_INET;
194         addr.sin_port = htons(port);
195         memcpy(&addr.sin_addr, hp->h_addr, hp->h_length);
196         if(connect(srv, (struct sockaddr *) &addr, sizeof(struct sockaddr_in))) {
197                 close(srv);
198                 die("error: cannot connect host '%s'\n", host);
199         }
200         /* login */
201         if(password)
202                 snprintf(bufout, sizeof bufout,
203                         "PASS %s\r\nNICK %s\r\nUSER %s localhost %s :%s\r\n",
204                         password, nick, nick, host, nick);
205         else
206                 snprintf(bufout, sizeof bufout, "NICK %s\r\nUSER %s localhost %s :%s\r\n",
207                          nick, nick, host, nick);
208         write(srv, bufout, strlen(bufout));
209         snprintf(ping, sizeof ping, "PING %s\r\n", host);
210         channel[0] = 0;
211         setbuf(stdout, NULL); /* unbuffered stdout */
212
213         for(;;) { /* main loop */
214                 FD_ZERO(&rd);
215                 FD_SET(0, &rd);
216                 FD_SET(srv, &rd);
217                 tv.tv_sec = 120;
218                 tv.tv_usec = 0;
219                 i = select(srv + 1, &rd, 0, 0, &tv);
220                 if(i < 0) {
221                         if(errno == EINTR)
222                                 continue;
223                         die("error: error on select()\n");
224                 }
225                 else if(i == 0) {
226                         if(time(NULL) - trespond >= PINGTIMEOUT)
227                                 die("error: sic shutting down: parse timeout\n");
228                         write(srv, ping, strlen(ping));
229                         continue;
230                 }
231                 if(FD_ISSET(srv, &rd)) {
232                         if(readl(srv, sizeof bufin, bufin) == -1)
233                                 die("error: remote host closed connection\n");
234                         parsesrv(bufin);
235                         trespond = time(NULL);
236                 }
237                 if(FD_ISSET(0, &rd)) {
238                         if(readl(0, sizeof bufin, bufin) == -1)
239                                 die("error: broken pipe\n");
240                         parsein(bufin);
241                 }
242         }
243         return 0;
244 }