]> git.armaanb.net Git - sic.git/blob - sic.c
applied portions of Mark Edgars 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         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) {
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 - 1);
84                 return;
85         }
86         else
87                 snprintf(bufout, sizeof bufout, "%s\r\n", msg + 1);
88         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                 snprintf(bufout, sizeof bufout, "<%s> %s", usr, txt);
127                 printl(chan, bufout);
128         }
129         else if(strncmp("PING", cmd, 4) == 0 && txt != NULL) {
130                 snprintf(bufout, sizeof bufout, "PONG %s\r\n", txt);
131                 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 && txt != NULL)
140                         strncpy(nick, txt, sizeof nick - 1);
141         }
142 }
143
144 int
145 readl(int fd, unsigned int len, char *buf) {
146         unsigned int i = 0;
147         char c = '\0';
148
149         do {
150                 if(read(fd, &c, sizeof(char)) != (ssize_t) 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(const int argc, char *const argv[]) {
162         int i;
163         struct timeval tv;
164         static struct addrinfo hints, *res, *r;
165         char ping[256];
166         fd_set rd;
167         char *password = NULL;
168
169         strncpy(nick, getenv("USER"), sizeof nick - 1);
170         for(i = 1; i < argc; i++)
171                 if(strcmp(argv[i], "-h") == 0) {
172                         if(++i < argc) host = argv[i];
173                 }
174                 else if(strcmp(argv[i], "-p") == 0) {
175                         if(++i < argc) port = argv[i];
176                 }
177                 else if(strcmp(argv[i], "-n") == 0) {
178                         if(++i < argc) strncpy(nick, argv[i], sizeof nick - 1);
179                 }
180                 else if(strcmp(argv[i], "-k") == 0) {
181                         if(++i < argc) password = argv[i];
182                 }
183                 else if(strcmp(argv[i], "-v") == 0)
184                         die("sic-%s, © 2005-2009 sic engineers\n", VERSION);
185                 else
186                         die("usage: sic [-h host] [-p port] [-n nick] [-k keyword] [-v]\n");
187
188         /* init */
189         memset(&hints, 0, sizeof hints);
190         hints.ai_family = AF_UNSPEC;
191         hints.ai_socktype = SOCK_STREAM;
192         if(getaddrinfo(host, port, &hints, &res) != 0)
193                 die("error: cannot resolve hostname '%s'\n", host);
194         for(r = res; r; r = r->ai_next) {
195                 if((srv = socket(r->ai_family, r->ai_socktype, r->ai_protocol)) == -1)
196                         continue;
197                 if(connect(srv, r->ai_addr, r->ai_addrlen) == 0)
198                         break;
199                 close(srv);
200         }
201         freeaddrinfo(res);
202         if(!r)
203                 die("error: cannot connect to host '%s'\n", host);
204
205         /* login */
206         if (password)
207                 snprintf(bufout, sizeof bufout,
208                         "PASS %s\r\nNICK %s\r\nUSER %s localhost %s :%s\r\n",
209                         password, nick, nick, host, nick);
210         else
211                 snprintf(bufout, sizeof bufout, "NICK %s\r\nUSER %s localhost %s :%s\r\n",
212                          nick, nick, host, nick);
213         write(srv, bufout, strlen(bufout));
214         snprintf(ping, sizeof ping, "PING %s\r\n", host);
215         channel[0] = '\0';
216         setbuf(stdout, NULL); /* unbuffered stdout */
217
218         for(;;) { /* main loop */
219                 FD_ZERO(&rd);
220                 FD_SET(0, &rd);
221                 FD_SET(srv, &rd);
222                 tv.tv_sec = 120;
223                 tv.tv_usec = 0;
224                 i = select(srv + 1, &rd, 0, 0, &tv);
225                 if(i < 0) {
226                         if(errno == EINTR)
227                                 continue;
228                         die("error: error on select()\n");
229                 }
230                 else if(i == 0) {
231                         if(time(NULL) - trespond >= PINGTIMEOUT)
232                                 die("error: sic shutting down: parse timeout\n");
233                         write(srv, ping, strlen(ping));
234                         continue;
235                 }
236                 if(FD_ISSET(srv, &rd) != 0) {
237                         if(readl(srv, sizeof bufin, bufin) == -1)
238                                 die("error: remote host closed connection\n");
239                         parsesrv(bufin);
240                         trespond = time(NULL);
241                 }
242                 if(FD_ISSET(0, &rd) != 0) {
243                         if(readl(0, sizeof bufin, bufin) == -1)
244                                 die("error: broken pipe\n");
245                         parsein(bufin);
246                 }
247         }
248         return 0;
249 }