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