irc.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. #include <ctype.h>
  2. #include <string.h>
  3. #ifdef __WIN32
  4. #define __USE_W32_SOCKETS
  5. #include <windows.h>
  6. #include <io.h>
  7. typedef int socklen_t;
  8. #else
  9. #include <sys/socket.h>
  10. #include <netinet/in.h>
  11. #include <sys/time.h>
  12. #include <unistd.h>
  13. #include <sys/ioctl.h>
  14. #include <netdb.h>
  15. #include <arpa/inet.h>
  16. #include <ctype.h>
  17. #ifndef SIOCGIFCONF
  18. #include <sys/sockio.h> // SIOCGIFCONF on Solaris, maybe others? [Shinomori]
  19. #endif
  20. #endif
  21. #include "../common/core.h"
  22. #include "../common/socket.h"
  23. #include "../common/malloc.h"
  24. #include "../common/db.h"
  25. #include "../common/timer.h"
  26. #include "../common/strlib.h"
  27. #include "../common/mmo.h"
  28. #include "../common/showmsg.h"
  29. #include "../common/version.h"
  30. #include "nullpo.h"
  31. #include "map.h"
  32. #include "pc.h"
  33. #include "irc.h"
  34. #include "intif.h" //For GM Broadcast [Zido]
  35. short use_irc=0;
  36. short irc_announce_flag=1;
  37. short irc_announce_mvp_flag=1;
  38. short irc_announce_jobchange_flag=1;
  39. short irc_announce_shop_flag=1;
  40. IRC_SI *irc_si=NULL;
  41. char irc_nick[30]="";
  42. char irc_password[32]="";
  43. char irc_channel[32]="";
  44. char irc_trade_channel[32]="";
  45. unsigned char irc_ip_str[128]="";
  46. unsigned long irc_ip=6667;
  47. unsigned short irc_port = 6667;
  48. int irc_fd=0;
  49. struct channel_data cd;
  50. int last_cd_user=0;
  51. int irc_connect_timer(int tid, unsigned int tick, int id, int data)
  52. {
  53. if(irc_si && session[irc_si->fd])
  54. return 0;
  55. irc_fd = make_connection(irc_ip,irc_port);
  56. if(irc_fd > 0){
  57. session[irc_fd]->func_parse = irc_parse;
  58. }
  59. return 0;
  60. }
  61. void irc_announce(char *buf)
  62. {
  63. char send_string[256];
  64. memset(send_string,'\0',256);
  65. sprintf(send_string,"PRIVMSG %s :%s",irc_channel, buf);
  66. irc_send(send_string);
  67. }
  68. void irc_announce_jobchange(struct map_session_data *sd)
  69. {
  70. char send_string[256];
  71. nullpo_retv(sd);
  72. memset(send_string,'\0',256);
  73. sprintf(send_string,"PRIVMSG %s :%s has changed into a %s.",irc_channel,sd->status.name,job_name(sd->status.class_));
  74. irc_send(send_string);
  75. }
  76. void irc_announce_shop(struct map_session_data *sd, int flag)
  77. {
  78. char send_string[256];
  79. char mapname[16];
  80. int maplen = 0;
  81. nullpo_retv(sd);
  82. memset(send_string,'\0',256);
  83. memset(mapname,'\0',16);
  84. if(flag){
  85. strcpy(mapname, map[sd->bl.m].name);
  86. maplen = strcspn(mapname,".");
  87. mapname[maplen] = '\0';
  88. mapname[0]=toupper(mapname[0]);
  89. sprintf(send_string,"PRIVMSG %s :%s has opened a shop, %s, at <%d,%d> in %s.",irc_trade_channel,sd->status.name,sd->message,sd->bl.x,sd->bl.y,mapname);
  90. } else
  91. sprintf(send_string,"PRIVMSG %s :%s has closed their shop.",irc_trade_channel,sd->status.name);
  92. irc_send(send_string);
  93. }
  94. void irc_announce_mvp(struct map_session_data *sd, struct mob_data *md)
  95. {
  96. char send_string[256];
  97. char mapname[16];
  98. int maplen = 0;
  99. nullpo_retv(sd);
  100. nullpo_retv(md);
  101. memset(send_string,'\0',256);
  102. memset(mapname,'\0',16);
  103. mapname[16]='\0';
  104. strcpy(mapname, map[md->bl.m].name);
  105. maplen = strcspn(mapname,".");
  106. mapname[maplen] = '\0';
  107. mapname[0]=toupper(mapname[0]);
  108. sprintf(send_string,"PRIVMSG %s :%s the %s has MVP'd %s in %s.",irc_channel,sd->status.name,job_name(sd->status.class_),md->name, mapname);
  109. irc_send(send_string);
  110. }
  111. int irc_parse(int fd)
  112. {
  113. if (session[fd]->eof){
  114. do_close(fd);
  115. irc_si = NULL;
  116. add_timer(gettick() + 15000, irc_connect_timer, 0, 0);
  117. return 0;
  118. }
  119. if (session[fd]->session_data == NULL){
  120. irc_si = (struct IRC_Session_Info*)aCalloc(1, sizeof(struct IRC_Session_Info));
  121. irc_si->fd = fd;
  122. irc_si->state = 0;
  123. session[fd]->session_data = irc_si;
  124. } else if (!irc_si) {
  125. irc_si = (struct IRC_Session_Info*)session[fd]->session_data;
  126. irc_si->fd = fd;
  127. }
  128. if(RFIFOREST(fd) > 0){
  129. char *incoming_string=aCalloc(RFIFOREST(fd),sizeof(char));
  130. memcpy(incoming_string,RFIFOP(fd,0),RFIFOREST(fd));
  131. send_to_parser(fd,incoming_string,"\n");
  132. RFIFOSKIP(fd,RFIFOREST(fd));
  133. aFree(incoming_string);
  134. }
  135. return 0;
  136. }
  137. int irc_keepalive_timer(int tid, unsigned int tick, int id, int data)
  138. {
  139. char send_string[128];
  140. memset(send_string,'\0',128);
  141. sprintf(send_string,"PRIVMSG %s : ", irc_nick);
  142. irc_send(send_string);
  143. add_timer(gettick() + 30000, irc_keepalive_timer, 0, 0);
  144. return 0;
  145. }
  146. void irc_send_sub(int fd, char transmit[4096])
  147. {
  148. sprintf(transmit,"%s%c",transmit,10);
  149. WFIFOHEAD(fd,strlen(transmit));
  150. memcpy(WFIFOP(fd,0),transmit, strlen(transmit));
  151. WFIFOSET(fd,strlen(transmit));
  152. }
  153. void irc_send(char *buf)
  154. {
  155. char transmit[4096];
  156. if(!irc_si || !session[irc_si->fd])
  157. return;
  158. memset(transmit,'\0',4096);
  159. sprintf(transmit,buf);
  160. irc_send_sub(irc_si->fd,transmit);
  161. }
  162. void irc_parse_sub(int fd, char *incoming_string)
  163. {
  164. char source[256];
  165. char command[256];
  166. char target[256];
  167. char message[8192];
  168. char send_string[8192];
  169. char *source_nick=NULL;
  170. char *source_ident=NULL;
  171. char *source_host=NULL;
  172. char *state_mgr=NULL;
  173. char cmd1[256];
  174. char cmd2[256];
  175. memset(source,'\0',256);
  176. memset(command,'\0',256);
  177. memset(target,'\0',256);
  178. memset(message,'\0',8192);
  179. memset(send_string,'\0',8192);
  180. memset(cmd1,'\0',256);
  181. memset(cmd2,'\0',256);
  182. sscanf(incoming_string, ":%255s %255s %255s :%4095[^\r\n]", source, command, target, message);
  183. if (source != NULL) {
  184. if (strstr(source,"!") != NULL) {
  185. source_nick = strtok_r(source,"!",&state_mgr);
  186. source_ident = strtok_r(NULL,"@",&state_mgr);
  187. source_host = strtok_r(NULL,"%%",&state_mgr);
  188. }
  189. }
  190. if (irc_si->state == 0){
  191. sprintf(send_string, "NICK %s", irc_nick);
  192. irc_send(send_string);
  193. sprintf(send_string, "USER eABot 8 * : eABot");
  194. irc_send(send_string);
  195. irc_si->state = 1;
  196. }
  197. else if (irc_si->state == 1){
  198. if(!strcmp(command,"001")){
  199. ShowStatus("IRC: Connected to IRC.\n");
  200. sprintf(send_string, "PRIVMSG nickserv :identify %s", irc_password);
  201. irc_send(send_string);
  202. sprintf(send_string, "JOIN %s", irc_channel);
  203. irc_send(send_string);
  204. irc_si->state = 2;
  205. }
  206. else if(!strcmp(command,"433")){
  207. ShowError("IRC: Nickname %s is already taken, IRC Client unable to connect.\n", irc_nick);
  208. sprintf(send_string, "QUIT");
  209. irc_send(send_string);
  210. if(session[fd])
  211. session[fd]->eof=1;
  212. }
  213. }
  214. else if (irc_si->state == 2){
  215. if(!strcmp(source, "PING")){
  216. sprintf(send_string, "PONG %s", command);
  217. irc_send(send_string);
  218. }
  219. else if((strcmpi(target,irc_channel)==0)||(strcmpi(target,irc_channel+1)==0)) {
  220. // Broadcast [Zido] (Work in Progress)
  221. if((strcmpi(command,"privmsg")==0)&&(sscanf(message,"@kami %255[^\r\n]",cmd1)>0)&&(target[0]=='#')) {
  222. if(get_access(source_nick)<ACCESS_OP)
  223. sprintf(send_string,"NOTICE %s :Access Denied",source_nick);
  224. else {
  225. sprintf(send_string,"%s: %s",source_nick,cmd1);
  226. intif_GMmessage(send_string,strlen(send_string)+1,0);
  227. sprintf(send_string,"NOTICE %s :Message Sent",source_nick);
  228. }
  229. irc_send(send_string);
  230. }
  231. // Refresh Names [Zido]
  232. else if((strcmpi(command,"join")==0)||(strcmpi(command,"part")==0)||(strcmpi(command,"mode")==0)||(strcmpi(command,"nick")==0)) {
  233. ShowInfo("IRC: Refreshing User List");
  234. irc_rmnames();
  235. printf("...");
  236. sprintf(send_string,"NAMES %s",irc_channel);
  237. printf("...");
  238. irc_send(send_string);
  239. printf("Done\n");
  240. }
  241. }
  242. // Names Reply [Zido]
  243. else if((strcmpi(command,"353")==0)) {
  244. ShowInfo("IRC: NAMES recieved\n");
  245. parse_names_packet(incoming_string);
  246. }
  247. }
  248. return;
  249. }
  250. int send_to_parser(int fd, char *input,char key[2])
  251. {
  252. char *temp_string=NULL;
  253. char *state_mgr=NULL;
  254. int total_loops=0;
  255. temp_string = strtok_r(input,key,&state_mgr);
  256. while (temp_string != NULL){
  257. total_loops = total_loops+1;
  258. irc_parse_sub(fd,temp_string);
  259. temp_string = strtok_r(NULL,key,&state_mgr);
  260. }
  261. return total_loops;
  262. }
  263. void do_final_irc(void)
  264. {
  265. }
  266. void do_init_irc(void)
  267. {
  268. struct hostent *irc_hostname;
  269. if(!use_irc)
  270. return;
  271. if (irc_ip_str[strlen(irc_ip_str)-1] == '\n')
  272. irc_ip_str[strlen(irc_ip_str)-1] = '\0';
  273. irc_hostname=gethostbyname(irc_ip_str);
  274. irc_ip_str[0]='\0';
  275. sprintf(irc_ip_str, "%d.%d.%d.%d", (unsigned char)irc_hostname->h_addr[0], (unsigned char)irc_hostname->h_addr[1],
  276. (unsigned char)irc_hostname->h_addr[2], (unsigned char)irc_hostname->h_addr[3]);
  277. irc_ip=inet_addr(irc_ip_str);
  278. irc_connect_timer(0, 0, 0, 0);
  279. add_timer_func_list(irc_connect_timer, "irc_connect_timer");
  280. add_timer_func_list(irc_keepalive_timer, "irc_keepalive_timer");
  281. add_timer(gettick() + 30000, irc_keepalive_timer, 0, 0);
  282. }
  283. //NAMES Packet(353) parser [Zido]
  284. int parse_names_packet(char *str) {
  285. char *tok;
  286. char source[256];
  287. char numeric[10];
  288. char target[256];
  289. char channel[256];
  290. char names[1024];
  291. memset(source,'\0',256);
  292. memset(numeric,'\0',10);
  293. memset(target,'\0',256);
  294. memset(channel,'\0',256);
  295. memset(names,'\0',1024);
  296. tok=strtok(str,"\r\n");
  297. sscanf(tok,":%255s %10s %255s = %255s :%1023[^\r\n]",source,numeric,target,channel,names);
  298. if(strcmpi(numeric,"353")==0)
  299. parse_names(names);
  300. while((tok=strtok(NULL,"\r\n"))!=NULL) {
  301. sscanf(tok,":%255s %10s %255s = %255s :%1023[^\r\n]",source,numeric,target,channel,names);
  302. if(strcmpi(numeric,"353")==0)
  303. parse_names(names);
  304. }
  305. return 0;
  306. }
  307. //User access level prefix parser [Zido]
  308. int parse_names(char *str) {
  309. char *tok;
  310. tok=strtok(str," ");
  311. switch(tok[0]) {
  312. case '~':
  313. set_access(tok+1,ACCESS_OWNER);
  314. break;
  315. case '&':
  316. set_access(tok+1,ACCESS_SOP);
  317. break;
  318. case '@':
  319. set_access(tok+1,ACCESS_OP);
  320. break;
  321. case '%':
  322. set_access(tok+1,ACCESS_HOP);
  323. break;
  324. case '+':
  325. set_access(tok+1,ACCESS_VOICE);
  326. break;
  327. default:
  328. set_access(tok,ACCESS_NORM);
  329. break;
  330. }
  331. while((tok=strtok(NULL," "))!=NULL) {
  332. switch(tok[0]) {
  333. case '~':
  334. set_access(tok+1,ACCESS_OWNER);
  335. break;
  336. case '&':
  337. set_access(tok+1,ACCESS_SOP);
  338. break;
  339. case '@':
  340. set_access(tok+1,ACCESS_OP);
  341. break;
  342. case '%':
  343. set_access(tok+1,ACCESS_HOP);
  344. break;
  345. case '+':
  346. set_access(tok+1,ACCESS_VOICE);
  347. break;
  348. default:
  349. set_access(tok,ACCESS_NORM);
  350. break;
  351. }
  352. }
  353. return 0;
  354. }
  355. //Store user's access level [Zido]
  356. int set_access(char *nick,int newlevel) {
  357. int i=0;
  358. for(i=0;i<=MAX_CHANNEL_USERS;i++) {
  359. if(strcmpi(cd.user[i].name,nick)==0) {
  360. cd.user[i].level=newlevel;
  361. return 1;
  362. }
  363. }
  364. strcpy(cd.user[last_cd_user].name,nick);
  365. cd.user[last_cd_user].level=newlevel;
  366. last_cd_user++;
  367. return 0;
  368. }
  369. //Returns users access level [Zido]
  370. int get_access(char *nick) {
  371. int i=0;
  372. for(i=0;i<=MAX_CHANNEL_USERS;i++) {
  373. if(strcmpi(cd.user[i].name,nick)==0) {
  374. return (cd.user[i].level);
  375. }
  376. }
  377. return -1;
  378. }
  379. int irc_rmnames() {
  380. int i=0;
  381. for(i=0;i<=MAX_CHANNEL_USERS;i++) {
  382. //memset(cd.user[i].name,'\0',256);
  383. cd.user[i].level=0;
  384. }
  385. last_cd_user=0;
  386. return 0;
  387. }