]> git.armaanb.net Git - sic.git/blob - sic.c
applied Martin Kopta's splint patch
[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         (void) 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         (void) 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         (void) snprintf(bufout, sizeof bufout, "<%s> %s", nick, msg);
58         printl(channel, bufout);
59         (void) snprintf(bufout, sizeof bufout, "PRIVMSG %s :%s\r\n", channel, msg);
60         (void) 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                 (void) snprintf(bufout, sizeof bufout, "JOIN %s\r\n", msg + 3);
74         else if(strncmp(msg + 1, "l ", 2) == 0)
75                 (void) 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) {
77                 if((p = strchr(msg + 3, ' ')))
78                         *(p++) = '\0';
79                 privmsg(msg + 3, p);
80                 return;
81         }
82         else if(strncmp(msg + 1, "s ", 2) == 0) {
83                 strncpy(channel, msg + 3, sizeof channel);
84                 return;
85         }
86         else
87                 (void) snprintf(bufout, sizeof bufout, "%s\r\n", msg + 1);
88         (void) write(srv, bufout, strlen(bufout));
89 }
90
91 void
92 parsesrv(char *msg) {
93         char *chan, *cmd, *p, *txt, *usr; 
94
95         txt = NULL;
96         usr = host;
97         if(msg == NULL || *msg == '\0' )
98                 return;
99         if(msg[0] != ':')
100                 cmd = msg;
101         else {
102                 if(!(p = strchr(msg, ' ')))
103                         return;
104                 *p = '\0';
105                 usr = msg + 1;
106                 cmd = ++p;
107                 if((p = strchr(usr, '!')))
108                         *p = '\0';
109         }
110         for(p = cmd; *p != '\0'; p++) /* remove CRLFs */
111                 if(*p == '\r' || *p == '\n')
112                         *p = '\0';
113         if((p = strchr(cmd, ':'))) {
114                 *p = '\0';
115                 txt = ++p;
116         }
117         if(strncmp("PONG", cmd, 4) == 0)
118                 return;
119         if(strncmp("PRIVMSG", cmd, 7) == 0 && txt != NULL) {
120                 if(!(p = strchr(cmd, ' ')))
121                         return;
122                 *p = '\0';
123                 chan = ++p;
124                 for(; *p != '\0' && *p != ' '; p++);
125                 *p = '\0';
126                 (void) snprintf(bufout, sizeof bufout, "<%s> %s", usr, txt);
127                 printl(chan, bufout);
128         }
129         else if(strncmp("PING", cmd, 4) == 0 && txt != NULL) {
130                 (void) snprintf(bufout, sizeof bufout, "PONG %s\r\n", txt);
131                 (void) write(srv, bufout, strlen(bufout));
132         }
133         else {
134                 if (txt != NULL)
135                         (void) snprintf(bufout, sizeof bufout, ">< %s: %s", cmd, txt);
136                 else
137                         (void) snprintf(bufout, sizeof bufout, ">< %s: ", cmd);
138                 printl(usr, bufout);
139                 if(strncmp("NICK", cmd, 4) == 0 && strncmp(usr, nick, sizeof nick) == 0 &&
140                                 txt != NULL)
141                         (void) strncpy(nick, txt, sizeof nick);
142         }
143 }
144
145 int
146 readl(int fd, unsigned int len, char *buf) {
147         unsigned int i = 0;
148         char c = '\0';
149
150         do {
151                 if(read(fd, &c, sizeof(char)) != (ssize_t) sizeof(char))
152                         return -1;
153                 buf[i++] = c;
154         }
155         while(c != '\n' && i < len);
156         buf[i - 1] = '\0';
157         return 0;
158 }
159
160
161 int
162 main(const int argc, char *const argv[]) {
163         int i;
164         struct timeval tv;
165         static struct addrinfo hints, *res, *r;
166         char ping[256];
167         fd_set rd;
168         char *password = NULL;
169
170         strncpy(nick, getenv("USER"), sizeof nick);
171         for(i = 1; i < argc; i++)
172                 if(strncmp(argv[i], "-h", 3) == 0) {
173                         if(++i < argc) host = argv[i];
174                 }
175                 else if(strncmp(argv[i], "-p", 3) == 0) {
176                         if(++i < argc) port = argv[i];
177                 }
178                 else if(strncmp(argv[i], "-n", 3) == 0) {
179                         if(++i < argc) strncpy(nick, argv[i], sizeof nick);
180                 }
181                 else if(strncmp(argv[i], "-k", 3) == 0) {
182                         if(++i < argc) password = argv[i];
183                 }
184                 else if(strncmp(argv[i], "-v", 3) == 0)
185                         die("sic-%s, © 2005-2009 sic engineers\n", VERSION);
186                 else
187                         die("usage: sic [-h host] [-p port] [-n nick] [-k keyword] [-v]\n");
188
189         /* init */
190         memset(&hints, 0, sizeof hints);
191         hints.ai_family = AF_UNSPEC;
192         hints.ai_socktype = SOCK_STREAM;
193         if(getaddrinfo(host, port, &hints, &res) != 0)
194                 die("error: cannot resolve hostname '%s'\n", host);
195         for(r = res; r; r = r->ai_next) {
196                 if((srv = socket(r->ai_family, r->ai_socktype, r->ai_protocol)) == -1)
197                         continue;
198                 if(connect(srv, r->ai_addr, r->ai_addrlen) == 0)
199                         break;
200                 (void) close(srv);
201         }
202         freeaddrinfo(res);
203         if(!r)
204                 die("error: cannot connect to host '%s'\n", host);
205
206         /* login */
207         if (password)
208                 (void) snprintf(bufout, sizeof bufout,
209                         "PASS %s\r\nNICK %s\r\nUSER %s localhost %s :%s\r\n",
210                         password, nick, nick, host, nick);
211         else
212                 (void) snprintf(bufout, sizeof bufout, "NICK %s\r\nUSER %s localhost %s :%s\r\n",
213                          nick, nick, host, nick);
214         (void) write(srv, bufout, strlen(bufout));
215         (void) snprintf(ping, sizeof ping, "PING %s\r\n", host);
216         channel[0] = '\0';
217         setbuf(stdout, NULL); /* unbuffered stdout */
218
219         for(;;) { /* main loop */
220                 FD_ZERO(&rd);
221                 FD_SET(0, &rd);
222                 FD_SET(srv, &rd);
223                 tv.tv_sec = 120;
224                 tv.tv_usec = 0;
225                 i = select(srv + 1, &rd, 0, 0, &tv);
226                 if(i < 0) {
227                         if(errno == EINTR)
228                                 continue;
229                         die("error: error on select()\n");
230                 }
231                 else if(i == 0) {
232                         if(time(NULL) - trespond >= PINGTIMEOUT)
233                                 die("error: sic shutting down: parse timeout\n");
234                         (void) write(srv, ping, strlen(ping));
235                         continue;
236                 }
237                 if(FD_ISSET(srv, &rd) != 0) {
238                         if(readl(srv, sizeof bufin, bufin) == -1)
239                                 die("error: remote host closed connection\n");
240                         parsesrv(bufin);
241                         trespond = time(NULL);
242                 }
243                 if(FD_ISSET(0, &rd) != 0) {
244                         if(readl(0, (unsigned int) sizeof bufin, bufin) == -1)
245                                 die("error: broken pipe\n");
246                         parsein(bufin);
247                 }
248         }
249         return 0;
250 }