]> jfr.im git - irc/evilnet/x3.git/blob - src/mod-snoop.c
tweaks
[irc/evilnet/x3.git] / src / mod-snoop.c
1 /* mod-snoop.c - User surveillance module (per pomac's spec)
2 * Copyright 2002-2004 srvx Development Team
3 *
4 * This file is part of x3.
5 *
6 * x3 is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with srvx; if not, write to the Free Software Foundation,
18 * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
19 */
20
21 /* Adds new section to srvx.conf:
22 * "modules" {
23 * "snoop" {
24 * // Where to send snoop messages?
25 * "channel" "#wherever";
26 * // Which bot?
27 * "bot" "OpServ";
28 * // Show new users and joins from net joins? (off by default)
29 * "show_bursts" "0";
30 * };
31 * };
32 */
33
34 #include "conf.h"
35 #include "helpfile.h"
36 #include "nickserv.h"
37
38 #ifdef HAVE_NETINET_IN_H
39 #include <netinet/in.h>
40 #endif
41 #ifdef HAVE_ARPA_INET_H
42 #include <arpa/inet.h>
43 #endif
44
45 extern time_t now;
46 static struct {
47 struct chanNode *channel;
48 struct userNode *bot;
49 unsigned int show_bursts : 1;
50 unsigned int enabled : 1;
51 } snoop_cfg;
52 static char timestamp[16];
53 const char *snoop_module_deps[] = { NULL };
54
55 static int finalized;
56 int snoop_finalize(void);
57
58 #define SNOOP(FORMAT, ARGS...) send_channel_message(snoop_cfg.channel, snoop_cfg.bot, "%s "FORMAT, timestamp , ## ARGS)
59 #define UPDATE_TIMESTAMP() strftime(timestamp, sizeof(timestamp), "[%H:%M:%S]", localtime(&now))
60
61 static void
62 snoop_nick_change(struct userNode *user, const char *old_nick) {
63 if (!snoop_cfg.enabled) return;
64 UPDATE_TIMESTAMP();
65 SNOOP("$bNICK$b change %s -> %s", old_nick, user->nick);
66 }
67
68 static int
69 snoop_join(struct modeNode *mNode) {
70 struct userNode *user = mNode->user;
71 struct chanNode *chan = mNode->channel;
72 if (!snoop_cfg.enabled) return 0;
73 if (user->uplink->burst && !snoop_cfg.show_bursts) return 0;
74 UPDATE_TIMESTAMP();
75 if (chan->members.used == 1) {
76 SNOOP("$bCREATE$b %s by %s", chan->name, user->nick);
77 } else {
78 SNOOP("$bJOIN$b %s by %s", chan->name, user->nick);
79 }
80 return 0;
81 }
82
83 static void
84 snoop_part(struct modeNode *mn, const char *reason) {
85 if (!snoop_cfg.enabled) return;
86 if (mn->user->dead) return;
87 UPDATE_TIMESTAMP();
88 SNOOP("$bPART$b %s by %s (%s)", mn->channel->name, mn->user->nick, reason ? reason : "");
89 }
90
91 static void
92 snoop_kick(struct userNode *kicker, struct userNode *victim, struct chanNode *chan) {
93 if (!snoop_cfg.enabled) return;
94 UPDATE_TIMESTAMP();
95 SNOOP("$bKICK$b %s from %s by %s", victim->nick, chan->name, (kicker ? kicker->nick : "some server"));
96 }
97
98 static int
99 snoop_new_user(struct userNode *user) {
100 if (!snoop_cfg.enabled) return 0;
101 if (user->uplink->burst && !snoop_cfg.show_bursts) return 0;
102 UPDATE_TIMESTAMP();
103 SNOOP("$bNICK$b %s %s@%s (%s) [%s] on %s", user->nick, user->ident, user->hostname, user->handle_info?user->handle_info->handle:"", irc_ntoa(&user->ip), user->uplink->name);
104 return 0;
105 }
106
107 static void
108 snoop_del_user(struct userNode *user, struct userNode *killer, const char *why) {
109 if (!snoop_cfg.enabled) return;
110 UPDATE_TIMESTAMP();
111 if (killer) {
112 SNOOP("$bKILL$b %s (%s@%s, on %s) by %s (%s)", user->nick, user->ident, user->hostname, user->uplink->name, killer->nick, why);
113 } else {
114 SNOOP("$bQUIT$b %s (%s@%s, on %s) (%s)", user->nick, user->ident, user->hostname, user->uplink->name, why);
115 }
116 }
117
118 static void
119 snoop_auth(struct userNode *user, UNUSED_ARG(struct handle_info *old_handle)) {
120 if (!snoop_cfg.enabled) return;
121 if (user->uplink->burst && !snoop_cfg.show_bursts) return;
122 if (user->handle_info) {
123 UPDATE_TIMESTAMP();
124 SNOOP("$bAUTH$b %s as %s", user->nick, user->handle_info->handle);
125 }
126 }
127
128 static void
129 snoop_user_mode(struct userNode *user, const char *mode_change) {
130 if (!snoop_cfg.enabled) return;
131 if (user->uplink->burst && !snoop_cfg.show_bursts) return;
132 if (!mode_change[1]) return; /* warning there has to be atleast one char in the buffer */
133 UPDATE_TIMESTAMP();
134 SNOOP("$bUMODE$b %s %s", user->nick, mode_change);
135 }
136
137 static void
138 snoop_oper(struct userNode *user) {
139 if (!snoop_cfg.enabled) return;
140 if (user->uplink->burst && !snoop_cfg.show_bursts) return;
141 UPDATE_TIMESTAMP();
142 SNOOP("$bOPER$b %s!%s@%s [%s] on %s", user->nick, user->ident, user->hostname, irc_ntoa(&user->ip), user->uplink->name);
143 }
144
145 static void
146 snoop_channel_mode(struct userNode *who, struct chanNode *channel, char **modes, unsigned int argc)
147 {
148 if (!snoop_cfg.enabled) return;
149 if(who) {
150 if (who->uplink->burst && !snoop_cfg.show_bursts) {
151 return;
152 }
153 } else {
154 return; /* Dont show X3 etc modes */
155 }
156
157 static char targets[MAXLEN], string[MAXLEN];
158 struct userNode *un = NULL;
159 char *tmp = NULL, *tg = NULL, *md = NULL;
160 int add = 0;
161
162 string[0] = 0;
163 targets[0] = 0;
164
165 if (argc > 0)
166 unsplit_string(modes, argc, string);
167 else
168 strcpy(string, *modes);
169
170 if((tg = strchr(string, ' ')))
171 {
172 *tg++ = 0;
173 for(md = string; *md; md++)
174 {
175 if (*md == '+')
176 {
177 add = 1;
178 md++;
179 }
180 if (*md == '-')
181 {
182 add = 0;
183 md++;
184 }
185 switch(*md)
186 {
187 case 'k':
188 {
189 strcat(targets, " ");
190 if ((tmp = strchr(tg, ' ')))
191 *tmp++ = 0;
192 strcat(targets, tg);
193 if(tmp)
194 tg = tmp;
195 break;
196 }
197 case 'l':
198 {
199 if(add)
200 {
201 strcat(targets, " ");
202 if ((tmp = strchr(tg, ' ')))
203 *tmp++ = 0;
204 strcat(targets, tg);
205 if(tmp)
206 tg = tmp;
207 break;
208 }
209 }
210 case 'b':
211 {
212 strcat(targets, " ");
213 if ((tmp = strchr(tg, ' ')))
214 *tmp++ = 0;
215 strcat(targets, tg);
216 if(tmp)
217 tg = tmp;
218 break;
219 }
220 case 'e':
221 {
222 strcat(targets, " ");
223 if ((tmp = strchr(tg, ' ')))
224 *tmp++ = 0;
225 strcat(targets, tg);
226 if(tmp)
227 tg = tmp;
228 break;
229 }
230 case 'o':
231 {
232 strcat(targets, " ");
233 if ((tmp = strchr(tg, ' ')))
234 *tmp++ = 0;
235 if((un = GetUserN(tg)))
236 strcat(targets, un->nick);
237 else
238 strcat(targets, tg);
239 if(tmp)
240 tg = tmp;
241 break;
242 }
243 case 'v':
244 {
245 strcat(targets, " ");
246 if ((tmp = strchr(tg, ' ')))
247 *tmp++ = 0;
248 if((un = GetUserN(tg)))
249 strcat(targets, un->nick);
250 else
251 strcat(targets, tg);
252 if(tmp)
253 tg = tmp;
254 break;
255 }
256 }
257 }
258 }
259 UPDATE_TIMESTAMP();
260 if (who)
261 SNOOP("$bMODE$b %s %s%s by %s", channel->name, string, targets, who->nick);
262 else
263 SNOOP("$bMODE$b %s %s%s", channel->name, string, targets);
264 }
265
266 static void
267 snoop_conf_read(void) {
268 dict_t node;
269 char *str;
270
271 node = conf_get_data("modules/snoop", RECDB_OBJECT);
272 if (!node)
273 return;
274 str = database_get_data(node, "channel", RECDB_QSTRING);
275 if (!str)
276 return;
277 snoop_cfg.channel = AddChannel(str, now, "+sntim", NULL, NULL);
278 if (!snoop_cfg.channel)
279 return;
280 str = database_get_data(node, "show_bursts", RECDB_QSTRING);
281 snoop_cfg.show_bursts = str ? enabled_string(str) : 0;
282 snoop_cfg.enabled = 1;
283 if (finalized)
284 snoop_finalize();
285 }
286
287 void
288 snoop_cleanup(void) {
289 snoop_cfg.enabled = 0;
290 unreg_del_user_func(snoop_del_user);
291 }
292
293 int
294 snoop_init(void) {
295 reg_exit_func(snoop_cleanup);
296 conf_register_reload(snoop_conf_read);
297 reg_nick_change_func(snoop_nick_change);
298 reg_join_func(snoop_join);
299 reg_part_func(snoop_part);
300 reg_kick_func(snoop_kick);
301 reg_new_user_func(snoop_new_user);
302 reg_del_user_func(snoop_del_user);
303 reg_auth_func(snoop_auth);
304 reg_channel_mode_func(snoop_channel_mode);
305 reg_user_mode_func(snoop_user_mode);
306 reg_oper_func(snoop_oper);
307
308 return 1;
309 }
310
311 int
312 snoop_finalize(void) {
313 struct mod_chanmode change;
314 dict_t node;
315 char *str;
316
317 finalized = 1;
318 node = conf_get_data("modules/snoop", RECDB_OBJECT);
319 if (!node)
320 return 0;
321 str = database_get_data(node, "bot", RECDB_QSTRING);
322 if (!str)
323 return 0;
324 snoop_cfg.bot = GetUserH(str);
325 if (!snoop_cfg.bot)
326 return 0;
327 mod_chanmode_init(&change);
328 change.argc = 1;
329 change.args[0].mode = MODE_CHANOP;
330 change.args[0].u.member = AddChannelUser(snoop_cfg.bot, snoop_cfg.channel);
331 mod_chanmode_announce(snoop_cfg.bot, snoop_cfg.channel, &change);
332 return 1;
333 }