sic.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. /* See LICENSE file for license details. */
  2. #include <sys/select.h>
  3. #include <ctype.h>
  4. #include <errno.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 "arg.h"
  12. #include "config.h"
  13. char *argv0;
  14. static char *host = DEFAULT_HOST;
  15. static char *port = DEFAULT_PORT;
  16. static char *password;
  17. static char nick[32];
  18. static char bufin[4096];
  19. static char bufout[4096];
  20. static char channel[256];
  21. static time_t trespond;
  22. static FILE *srv;
  23. #undef strlcpy
  24. #include "strlcpy.c"
  25. #include "util.c"
  26. static void
  27. pout(char *channel, char *fmt, ...) {
  28. static char timestr[80];
  29. time_t t;
  30. va_list ap;
  31. va_start(ap, fmt);
  32. vsnprintf(bufout, sizeof bufout, fmt, ap);
  33. va_end(ap);
  34. t = time(NULL);
  35. strftime(timestr, sizeof timestr, TIMESTAMP_FORMAT, localtime(&t));
  36. fprintf(stdout, "%-12s: %s %s\n", channel, timestr, bufout);
  37. }
  38. static void
  39. sout(char *fmt, ...) {
  40. va_list ap;
  41. va_start(ap, fmt);
  42. vsnprintf(bufout, sizeof bufout, fmt, ap);
  43. va_end(ap);
  44. fprintf(srv, "%s\r\n", bufout);
  45. }
  46. static void
  47. privmsg(char *channel, char *msg) {
  48. if(channel[0] == '\0') {
  49. pout("", "No channel to send to");
  50. return;
  51. }
  52. pout(channel, "<%s> %s", nick, msg);
  53. sout("PRIVMSG %s :%s", channel, msg);
  54. }
  55. static void
  56. parsein(char *s) {
  57. char c, *p;
  58. if(s[0] == '\0')
  59. return;
  60. skip(s, '\n');
  61. if(s[0] != COMMAND_PREFIX_CHARACTER) {
  62. privmsg(channel, s);
  63. return;
  64. }
  65. c = *++s;
  66. if(c != '\0' && isspace(s[1])) {
  67. p = s + 2;
  68. switch(c) {
  69. case 'j':
  70. sout("JOIN %s", p);
  71. if(channel[0] == '\0')
  72. strlcpy(channel, p, sizeof channel);
  73. return;
  74. case 'l':
  75. s = eat(p, isspace, 1);
  76. p = eat(s, isspace, 0);
  77. if(!*s)
  78. s = channel;
  79. if(*p)
  80. *p++ = '\0';
  81. if(!*p)
  82. p = DEFAULT_PARTING_MESSAGE;
  83. sout("PART %s :%s", s, p);
  84. return;
  85. case 'm':
  86. s = eat(p, isspace, 1);
  87. p = eat(s, isspace, 0);
  88. if(*p)
  89. *p++ = '\0';
  90. privmsg(s, p);
  91. return;
  92. case 's':
  93. strlcpy(channel, p, sizeof channel);
  94. return;
  95. }
  96. }
  97. sout("%s", s);
  98. }
  99. static void
  100. parsesrv(char *cmd) {
  101. char *usr, *par, *txt;
  102. usr = host;
  103. if(!cmd || !*cmd)
  104. return;
  105. if(cmd[0] == ':') {
  106. usr = cmd + 1;
  107. cmd = skip(usr, ' ');
  108. if(cmd[0] == '\0')
  109. return;
  110. skip(usr, '!');
  111. }
  112. skip(cmd, '\r');
  113. par = skip(cmd, ' ');
  114. txt = skip(par, ':');
  115. trim(par);
  116. if(!strcmp("PONG", cmd))
  117. return;
  118. if(!strcmp("PRIVMSG", cmd))
  119. pout(par, "<%s> %s", usr, txt);
  120. else if(!strcmp("PING", cmd))
  121. sout("PONG %s", txt);
  122. else {
  123. pout(usr, ">< %s (%s): %s", cmd, par, txt);
  124. if(!strcmp("NICK", cmd) && !strcmp(usr, nick))
  125. strlcpy(nick, txt, sizeof nick);
  126. }
  127. }
  128. static void
  129. usage(void) {
  130. eprint("usage: sic [-h host] [-p port] [-n nick] [-k keyword] [-v]\n", argv0);
  131. }
  132. int
  133. main(int argc, char *argv[]) {
  134. struct timeval tv;
  135. const char *user = getenv("USER");
  136. int n;
  137. fd_set rd;
  138. strlcpy(nick, user ? user : "unknown", sizeof nick);
  139. ARGBEGIN {
  140. case 'h':
  141. host = EARGF(usage());
  142. break;
  143. case 'p':
  144. port = EARGF(usage());
  145. break;
  146. case 'n':
  147. strlcpy(nick, EARGF(usage()), sizeof nick);
  148. break;
  149. case 'k':
  150. password = EARGF(usage());
  151. break;
  152. case 'v':
  153. eprint("sic-"VERSION", © 2005-2014 Kris Maglione, Anselm R. Garbe, Nico Golde\n");
  154. break;
  155. default:
  156. usage();
  157. } ARGEND;
  158. /* init */
  159. srv = fdopen(dial(host, port), "r+");
  160. if (!srv)
  161. eprint("fdopen:");
  162. /* login */
  163. if(password)
  164. sout("PASS %s", password);
  165. sout("NICK %s", nick);
  166. sout("USER %s localhost %s :%s", nick, host, nick);
  167. fflush(srv);
  168. setbuf(stdout, NULL);
  169. setbuf(srv, NULL);
  170. setbuf(stdin, NULL);
  171. #ifdef __OpenBSD__
  172. if (pledge("stdio", NULL) == -1)
  173. eprint("error: pledge:");
  174. #endif
  175. for(;;) { /* main loop */
  176. FD_ZERO(&rd);
  177. FD_SET(0, &rd);
  178. FD_SET(fileno(srv), &rd);
  179. tv.tv_sec = 120;
  180. tv.tv_usec = 0;
  181. n = select(fileno(srv) + 1, &rd, 0, 0, &tv);
  182. if(n < 0) {
  183. if(errno == EINTR)
  184. continue;
  185. eprint("sic: error on select():");
  186. }
  187. else if(n == 0) {
  188. if(time(NULL) - trespond >= 300)
  189. eprint("sic shutting down: parse timeout\n");
  190. sout("PING %s", host);
  191. continue;
  192. }
  193. if(FD_ISSET(fileno(srv), &rd)) {
  194. if(fgets(bufin, sizeof bufin, srv) == NULL)
  195. eprint("sic: remote host closed connection\n");
  196. parsesrv(bufin);
  197. trespond = time(NULL);
  198. }
  199. if(FD_ISSET(0, &rd)) {
  200. if(fgets(bufin, sizeof bufin, stdin) == NULL)
  201. eprint("sic: broken pipe\n");
  202. parsein(bufin);
  203. }
  204. }
  205. return 0;
  206. }