ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/ermyth/modules/protocol/nefarious.C
Revision: 1.8
Committed: Sat Sep 22 14:27:28 2007 UTC (16 years, 8 months ago) by pippijn
Content type: text/plain
Branch: MAIN
CVS Tags: HEAD
Changes since 1.7: +8 -12 lines
Log Message:
split up ermyth into ermyth-modules, libermyth (currently just ermyth-util) and ermyth-core

File Contents

# Content
1 /**
2 * nefarious.C: This file contains protocol support for P10 ircd's.
3 *
4 * Copyright © 2007 Pippijn van Steenhoven / The Ermyth Team
5 * Rights to this code are as documented in COPYING.
6 *
7 *
8 * Portions of this file were derived from sources bearing the following license:
9 * Copyright © 2005-2006 William Pitcock, et al.
10 * Rights to this code are documented in doc/pod/license.pod.
11 *
12 * $Id: nefarious.C,v 1.7 2007-09-16 18:54:44 pippijn Exp $
13 */
14
15 #include <boost/foreach.hpp>
16
17 #include "atheme.h"
18 #include <util/time.h>
19 #include <libermyth.h>
20 #include "servers.h"
21 #include <account/myuser.h>
22 #include "uplink.h"
23 #include "pmodule.h"
24 #include "protocol/nefarious.h"
25
26 static char const rcsid[] = "$Id: nefarious.C,v 1.7 2007-09-16 18:54:44 pippijn Exp $";
27
28 /* *INDENT-OFF* */
29
30 namespace protocol
31 {
32 static ircd_t Nefarious = {
33 "Nefarious IRCU 0.4.0 or later", /* IRCd name */
34 "$", /* TLD Prefix, used by Global. */
35 true, /* Whether or not we use IRCNet/TS6 UID */
36 false, /* Whether or not we use RCOMMAND */
37 false, /* Whether or not we support channel owners. */
38 false, /* Whether or not we support channel protection. */
39 true, /* Whether or not we support halfops. */
40 true, /* Whether or not we use P10 */
41 true, /* Whether or not we use vhosts. */
42 CMODE_PERM|CMODE_OPERONLY|CMODE_ADMONLY, /* Oper-only cmodes */
43 0, /* Integer flag for owner channel flag. */
44 0, /* Integer flag for protect channel flag. */
45 CMODE_HALFOP, /* Integer flag for halfops. */
46 "+", /* Mode we set for owner. */
47 "+", /* Mode we set for protect. */
48 "+", /* Mode we set for halfops. */
49 PROTOCOL_NEFARIOUS, /* Protocol type */
50 CMODE_PERM, /* Permanent cmodes */
51 "be", /* Ban-like cmodes */
52 'e', /* Except mchar */
53 'e', /* Invex mchar (+e also exempts from +i in Nefarious) */
54 IRCD_CIDR_BANS /* Flags */
55 };
56
57 struct nefarious_handler : handler
58 {
59 nefarious_handler ();
60 virtual ~nefarious_handler ();
61
62 virtual unsigned int server_login (void);
63 virtual void introduce_nick (user_t *u);
64 virtual void invite_sts (user_t *source, user_t *target, channel_t *channel);
65 virtual void quit_sts (user_t *u, char const * const reason);
66 virtual void wallops_sts (char const * const text);
67 virtual void join_sts (channel_t *c, user_t *u, bool isnew, char const * const modes);
68 virtual void kick (char const * const from, char const * const channel, char const * const to, char const * const reason);
69 virtual void privmsg (char const * const from, char const * const target, char const * const fmt, ...);
70 virtual void notice_user_sts (user_t *from, user_t *target, char const * const text);
71 virtual void notice_global_sts (user_t *from, char const * const mask, char const * const text);
72 virtual void notice_channel_sts (user_t *from, channel_t *target, char const * const text);
73 virtual void wallchops (user_t *sender, channel_t *channel, char const * const message);
74 virtual void numeric_sts (char const * const from, int numeric, char const * const target, char const * const fmt, ...);
75 virtual void skill (char const * const from, char const * const nick, char const * const fmt, ...);
76 virtual void part_sts (channel_t *c, user_t *u);
77 virtual void kline_sts (char const * const server, char const * const user, char const * const host, long duration, char const * const reason);
78 virtual void unkline_sts (char const * const server, char const * const user, char const * const host);
79 virtual void topic_sts (channel_t *c, char const * const setter, time_t ts, time_t prevts, char const * const topic);
80 virtual void mode_sts (char const * const sender, channel_t *target, char const * const modes);
81 virtual void ping_sts (void);
82 virtual void ircd_on_login (char const * const origin, char const * const user, char const * const wantedhost);
83 virtual bool ircd_on_logout (char const * const origin, char const * const user, char const * const wantedhost);
84 virtual void jupe (char const * const server, char const * const reason);
85 };
86
87 static cmode_t nefarious_mode_list[] = {
88 { 'a', CMODE_ADMONLY },
89 { 'i', CMODE_INVITE },
90 { 'm', CMODE_MOD },
91 { 'n', CMODE_NOEXT },
92 { 'p', CMODE_PRIV },
93 { 'r', CMODE_REGONLY },
94 { 's', CMODE_SEC },
95 { 't', CMODE_TOPIC },
96 { 'z', CMODE_PERM },
97 { 'c', CMODE_NOCOLOR },
98 { 'C', CMODE_NOCTCP },
99 { 'D', CMODE_DELAYED },
100 { 'Q', CMODE_NOQUIT },
101 { 'N', CMODE_NONOTICE },
102 { 'M', CMODE_SOFTMOD },
103 { 'C', CMODE_NOCTCP },
104 { 'S', CMODE_STRIP },
105 { 'T', CMODE_NOAMSG },
106 { 'O', CMODE_OPERONLY },
107 { 'L', CMODE_SOFTPRIV },
108 { 'Z', CMODE_SSLONLY },
109 { '\0', 0 }
110 };
111
112 static extmode_t nefarious_ignore_mode_list[] = {
113 { '\0', 0 }
114 };
115
116 static cmode_t nefarious_status_mode_list[] = {
117 { 'o', CMODE_OP },
118 { 'h', CMODE_HALFOP },
119 { 'v', CMODE_VOICE },
120 { '\0', 0 }
121 };
122
123 static cmode_t nefarious_prefix_mode_list[] = {
124 { '@', CMODE_OP },
125 { '%', CMODE_HALFOP },
126 { '+', CMODE_VOICE },
127 { '\0', 0 }
128 };
129
130 static void check_hidehost(user_t *u);
131
132 /* *INDENT-ON* */
133
134 /* login to our uplink */
135 unsigned int
136 nefarious_handler::server_login (void)
137 {
138 int ret;
139
140 ret = sts ("PASS :%s", curr_uplink->pass);
141 if (ret == 1)
142 return 1;
143
144 me.bursting = true;
145
146 /* SERVER irc.undernet.org 1 933022556 947908144 J10 AA]]] :[127.0.0.1] A Undernet Server */
147 sts ("SERVER %s 1 %ld %ld J10 %s]]] +s :%s", me.name, me.start, NOW, me.numeric, me.desc);
148
149 services_init ();
150
151 sts ("%s EB", me.numeric);
152
153 return 0;
154 }
155
156 /* introduce a client */
157 void
158 nefarious_handler::introduce_nick (user_t *u)
159 {
160 sts ("%s N %s 1 %ld %s %s +%s%sk ]]]]]] %s :%s", me.numeric, u->nick, u->ts, u->user, u->host, "io", chansvs.fantasy ? "" : "d", u->uid, u->gecos);
161 }
162
163 /* invite a user to a channel */
164 void
165 nefarious_handler::invite_sts (user_t *sender, user_t *target, channel_t *channel)
166 {
167 /* target is a nick, weird eh? -- jilles */
168 sts ("%s I %s %s", sender->uid, target->nick, channel->name);
169 }
170
171 void
172 nefarious_handler::quit_sts (user_t *u, char const * const reason)
173 {
174 if (!me.connected)
175 return;
176
177 sts ("%s Q :%s", u->uid, reason);
178 }
179
180 /* WALLOPS wrapper */
181 void
182 nefarious_handler::wallops_sts (char const * const text)
183 {
184 sts ("%s WA :%s", me.numeric, text);
185 }
186
187 /* join a channel */
188 void
189 nefarious_handler::join_sts (channel_t *c, user_t *u, bool isnew, char const * const modes)
190 {
191 /* If the channel doesn't exist, we need to create it. */
192 if (isnew)
193 {
194 sts ("%s C %s %ld", u->uid, c->name, c->ts);
195 if (modes[0] && modes[1])
196 sts ("%s M %s %s", u->uid, c->name, modes);
197 }
198 else
199 {
200 sts ("%s J %s %ld", u->uid, c->name, c->ts);
201 sts ("%s M %s +o %s", me.numeric, c->name, u->uid);
202 }
203 }
204
205 /* kicks a user from a channel */
206 void
207 nefarious_handler::kick (char const * const from, char const * const channel, char const * const to, char const * const reason)
208 {
209 channel_t *chan = channel_find (channel);
210 user_t *fptr = user_find_named (from);
211 user_t *user = user_find_named (to);
212
213 if (!chan || !user || !fptr)
214 return;
215
216 sts ("%s K %s %s :%s", fptr->uid, channel, user->uid, reason);
217
218 chanuser_delete (chan, user);
219 }
220
221 /* PRIVMSG wrapper */
222 void
223 nefarious_handler::privmsg (char const * const from, char const * const target, char const * const fmt, ...)
224 {
225 va_list ap;
226 user_t *u = user_find_named (from);
227 char buf[BUFSIZE];
228
229 if (!u)
230 return;
231
232 va_start (ap, fmt);
233 vsnprintf (buf, BUFSIZE, fmt, ap);
234 va_end (ap);
235
236 sts ("%s P %s :%s", u->uid, target, buf);
237 }
238
239 /* NOTICE wrapper */
240 void
241 nefarious_handler::notice_user_sts (user_t *from, user_t *target, char const * const text)
242 {
243 sts ("%s O %s :%s", from ? from->uid : me.numeric, target->uid, text);
244 }
245
246 void
247 nefarious_handler::notice_global_sts (user_t *from, char const * const mask, char const * const text)
248 {
249 if (!strcmp (mask, "*"))
250 foreach (tld_t *tld, tld_t::list)
251 sts ("%s O %s*%s :%s", from ? from->uid : me.numeric, ircd->tldprefix, tld->name, text);
252 else
253 sts ("%s O %s%s :%s", from ? from->uid : me.numeric, ircd->tldprefix, mask, text);
254 }
255
256 void
257 nefarious_handler::notice_channel_sts (user_t *from, channel_t *target, char const * const text)
258 {
259 sts ("%s O %s :%s", from ? from->uid : me.numeric, target->name, text);
260 }
261
262 void
263 nefarious_handler::wallchops (user_t *sender, channel_t *channel, char const * const message)
264 {
265 sts ("%s WC %s :%s", sender->uid, channel->name, message);
266 }
267
268 void
269 nefarious_handler::numeric_sts (char const * const from, int numeric, char const * const target, char const * const fmt, ...)
270 {
271 va_list ap;
272 char buf[BUFSIZE];
273 user_t *source_p, *target_p;
274
275 source_p = user_find_named (from);
276 target_p = user_find_named (target);
277
278 if (!target_p)
279 return;
280
281 va_start (ap, fmt);
282 vsnprintf (buf, BUFSIZE, fmt, ap);
283 va_end (ap);
284
285 sts ("%s %d %s %s", source_p ? source_p->uid : me.numeric, numeric, target_p->uid, buf);
286 }
287
288 /* KILL wrapper */
289 void
290 nefarious_handler::skill (char const * const from, char const * const nick, char const * const fmt, ...)
291 {
292 va_list ap;
293 char buf[BUFSIZE];
294 user_t *fptr = user_find_named (from);
295 user_t *tptr = user_find_named (nick);
296
297 if (!tptr)
298 return;
299
300 va_start (ap, fmt);
301 vsnprintf (buf, BUFSIZE, fmt, ap);
302 va_end (ap);
303
304 sts ("%s D %s :%s!%s!%s (%s)", fptr ? fptr->uid : me.numeric, tptr->uid, from, from, from, buf);
305 }
306
307 /* PART wrapper */
308 void
309 nefarious_handler::part_sts (channel_t *c, user_t *u)
310 {
311 sts ("%s L %s", u->uid, c->name);
312 }
313
314 /* server-to-server KLINE wrapper */
315 void
316 nefarious_handler::kline_sts (char const * const server, char const * const user, char const * const host, long duration, char const * const reason)
317 {
318 if (!me.connected)
319 return;
320
321 /* hold permanent akills for four weeks -- jilles */
322 sts ("%s GL * +%s@%s %ld :%s", me.numeric, user, host, duration > 0 ? duration : 2419200, reason);
323 }
324
325 /* server-to-server UNKLINE wrapper */
326 void
327 nefarious_handler::unkline_sts (char const * const server, char const * const user, char const * const host)
328 {
329 if (!me.connected)
330 return;
331
332 sts ("%s GL * -%s@%s", me.numeric, user, host);
333 }
334
335 /* topic wrapper */
336 void
337 nefarious_handler::topic_sts (channel_t *c, char const * const setter, time_t ts, time_t prevts, char const * const topic)
338 {
339 if (!me.connected || !c)
340 return;
341
342 /* for nefarious, set topicsetter iff we can set the proper topicTS */
343 if (ts > prevts || prevts == 0)
344 sts ("%s T %s %s %ld %ld :%s", chansvs.me->me->uid, c->name, setter, c->ts, ts, topic);
345 else
346 {
347 ts = NOW;
348 if (ts < prevts)
349 ts = prevts + 1;
350 sts ("%s T %s %ld %ld :%s", chansvs.me->me->uid, c->name, c->ts, NOW, topic);
351 c->topicts = ts;
352 }
353 }
354
355 /* mode wrapper */
356 void
357 nefarious_handler::mode_sts (char const * const sender, channel_t *target, char const * const modes)
358 {
359 user_t *fptr = user_find_named (sender);
360
361 if (!fptr)
362 return;
363
364 sts ("%s M %s %s", fptr->uid, target->name, modes);
365 }
366
367 /* ping wrapper */
368 void
369 nefarious_handler::ping_sts (void)
370 {
371 if (!me.connected)
372 return;
373
374 sts ("%s G !%ld %s %ld", me.numeric, NOW, me.name, NOW);
375 }
376
377 /* protocol-specific stuff to do on login */
378 void
379 nefarious_handler::ircd_on_login (char const * const origin, char const * const user, char const * const wantedhost)
380 {
381 user_t *u = user_find_named (origin);
382
383 if (!u)
384 return;
385
386 sts ("%s AC %s R %s", me.numeric, u->uid, u->myuser->name);
387 check_hidehost (u);
388 }
389
390 /* P10 does not support logout, so kill the user
391 * we can't keep track of which logins are stale and which aren't -- jilles
392 * Except we can in Nefarious --nenolod
393 */
394 bool
395 nefarious_handler::ircd_on_logout (char const * const origin, char const * const user, char const * const wantedhost)
396 {
397 user_t *u = user_find_named (origin);
398
399 if (!me.connected)
400 return false;
401
402 if (!u)
403 return false;
404
405 sts ("%s AC %s U", me.numeric, u->uid);
406 if (u->flags & UF_HIDEHOSTREQ && me.hidehostsuffix != NULL && !strcmp (u->vhost + strlen (u->vhost) - strlen (me.hidehostsuffix), me.hidehostsuffix))
407 {
408 slog (LG_DEBUG, "nefarious_ircd_on_logout(): removing +x vhost for %s: %s -> %s", u->nick, u->vhost, u->host);
409 strlcpy (u->vhost, u->host, sizeof u->vhost);
410 }
411
412 return false;
413 }
414
415 void
416 nefarious_handler::jupe (char const * const server, char const * const reason)
417 {
418 server_t *s;
419
420 if (!me.connected)
421 return;
422
423 /* hold it for a day (arbitrary) -- jilles */
424 /* get rid of local deactivation too */
425 s = server_find (server);
426 if (s != NULL && s->uplink != NULL)
427 sts ("%s JU %s +%s %d %ld :%s", me.numeric, s->uplink->sid, server, 86400, NOW, reason);
428 sts ("%s JU * +%s %d %ld :%s", me.numeric, server, 86400, NOW, reason);
429 }
430
431 static void
432 m_topic (sourceinfo_t *si, int parc, char *parv[])
433 {
434 channel_t *c = channel_find (parv[0]);
435 char *source;
436 time_t ts = 0;
437
438 if (!c)
439 return;
440
441 if (si->s != NULL)
442 source = si->s->name;
443 else
444 source = si->su->nick;
445
446 if (parc > 2)
447 ts = atoi (parv[parc - 2]);
448 if (ts == 0)
449 ts = NOW;
450 else if (c->topic != NULL && ts < c->topicts)
451 return;
452 handle_topic_from (si, c, parc > 4 ? parv[parc - 4] : source, ts, parv[parc - 1]);
453 }
454
455 /* AB G !1119920789.573932 services.atheme.org 1119920789.573932 */
456 static void
457 m_ping (sourceinfo_t *si, int parc, char *parv[])
458 {
459 /* reply to PING's */
460 sts ("%s Z %s %s %s", me.numeric, parv[0], parv[1], parv[2]);
461 }
462
463 static void
464 m_pong (sourceinfo_t *si, int parc, char *parv[])
465 {
466 me.uplinkpong = NOW;
467
468 /* -> :test.projectxero.net PONG test.projectxero.net :shrike.malkier.net */
469 if (me.bursting)
470 {
471 #ifdef HAVE_GETTIMEOFDAY
472 e_time (burstime, &burstime);
473
474 slog (LG_INFO, "m_pong(): finished synching with uplink (%d %s)", (tv2ms (&burstime) > 1000) ? (tv2ms (&burstime) / 1000) : tv2ms (&burstime), (tv2ms (&burstime) > 1000) ? "s" : "ms");
475
476 wallops ("Finished synching to network in %d %s.", (tv2ms (&burstime) > 1000) ? (tv2ms (&burstime) / 1000) : tv2ms (&burstime), (tv2ms (&burstime) > 1000) ? "s" : "ms");
477 #else
478 slog (LG_INFO, "m_pong(): finished synching with uplink");
479 wallops ("Finished synching to network.");
480 #endif
481
482 me.bursting = false;
483 }
484 }
485
486 static void
487 m_privmsg (sourceinfo_t *si, int parc, char *parv[])
488 {
489 if (parc != 2)
490 return;
491
492 handle_message (si, parv[0], false, parv[1]);
493 }
494
495 static void
496 m_notice (sourceinfo_t *si, int parc, char *parv[])
497 {
498 if (parc != 2)
499 return;
500
501 handle_message (si, parv[0], true, parv[1]);
502 }
503
504 static void
505 m_create (sourceinfo_t *si, int parc, char *parv[])
506 {
507 char buf[BUFSIZE];
508 int chanc;
509 char *chanv[256];
510 int i;
511
512 chanc = sjtoken (parv[0], ',', chanv);
513
514 for (i = 0; i < chanc; i++)
515 {
516 channel_t *c = channel_add (chanv[i], atoi (parv[1]), si->su->server);
517
518 /* Tell the core to check mode locks now,
519 * otherwise it may only happen after the next
520 * mode change.
521 * P10 does not allow any redundant modes
522 * so this will not look ugly. -- jilles */
523 channel_mode (NULL, c, "+");
524
525 buf[0] = '@';
526 buf[1] = '\0';
527
528 strlcat (buf, si->su->uid, BUFSIZE);
529
530 chanuser_add (c, buf);
531 }
532 }
533
534 static void
535 m_join (sourceinfo_t *si, int parc, char *parv[])
536 {
537 int chanc;
538 char *chanv[256];
539 int i;
540 node_t *n, *tn;
541 chanuser_t *cu;
542
543 /* JOIN 0 is really a part from all channels */
544 if (!strcmp (parv[0], "0"))
545 {
546 LIST_FOREACH_SAFE (n, tn, si->su->channels.head)
547 {
548 cu = (chanuser_t *) n->data;
549 chanuser_delete (cu->chan, si->su);
550 }
551 return;
552 }
553 if (parc < 2)
554 return;
555
556 chanc = sjtoken (parv[0], ',', chanv);
557
558 for (i = 0; i < chanc; i++)
559 {
560 channel_t *c = channel_find (chanv[i]);
561
562 if (!c)
563 {
564 c = channel_add (chanv[i], atoi (parv[1]), si->su->server);
565 channel_mode (NULL, c, "+");
566 }
567
568 chanuser_add (c, si->su->uid);
569 }
570 }
571
572 static void
573 m_burst (sourceinfo_t *si, int parc, char *parv[])
574 {
575 channel_t *c;
576 unsigned int modec;
577 char *modev[16];
578 unsigned int userc;
579 char *userv[256];
580 unsigned int i;
581 int j;
582 char prefix[16];
583 char newnick[16 + NICKLEN];
584 char *p;
585 time_t ts;
586 int bantype;
587
588 /* S BURST <channel> <ts> [parameters]
589 * parameters can be:
590 * +<simple mode>
591 * %<bans separated with spaces>
592 * <nicks>
593 */
594 ts = atoi (parv[1]);
595
596 c = channel_find (parv[0]);
597
598 if (c == NULL)
599 {
600 slog (LG_DEBUG, "m_burst(): new channel: %s", parv[0]);
601 c = channel_add (parv[0], ts, si->s);
602 }
603 else if (ts < c->ts)
604 {
605 chanuser_t *cu;
606 node_t *n;
607
608 clear_simple_modes (c);
609 chanban_clear (c);
610 handle_topic_from (si, c, "", 0, "");
611 LIST_FOREACH (n, c->members.head)
612 {
613 cu = (chanuser_t *) n->data;
614 if (cu->user->server == me.me)
615 {
616 /* it's a service, reop */
617 sts ("%s M %s +o %s", me.numeric, c->name, CLIENT_NAME (cu->user));
618 cu->modes = CMODE_OP;
619 }
620 else
621 cu->modes = 0;
622 }
623
624 slog (LG_DEBUG, "m_burst(): TS changed for %s (%ld -> %ld)", c->name, c->ts, ts);
625 c->ts = ts;
626 c->callback.tschange (c);
627 }
628 if (parc < 3 || parv[2][0] != '+')
629 {
630 /* Tell the core to check mode locks now,
631 * otherwise it may only happen after the next
632 * mode change. -- jilles */
633 channel_mode (NULL, c, "+");
634 }
635
636 bantype = 'b';
637 j = 2;
638 while (j < parc)
639 {
640 if (parv[j][0] == '+')
641 {
642 modec = 0;
643 modev[modec++] = parv[j++];
644 if (strchr (modev[0], 'k') && j < parc)
645 modev[modec++] = parv[j++];
646 if (strchr (modev[0], 'l') && j < parc)
647 modev[modec++] = parv[j++];
648 channel_mode (NULL, c, modec, modev);
649 }
650 else if (parv[j][0] == '%')
651 {
652 userc = sjtoken (parv[j++] + 1, ' ', userv);
653 for (i = 0; i < userc; i++)
654 if (!strcmp (userv[i], "~"))
655 /* A ban "~" means exceptions are
656 * following */
657 bantype = 'e';
658 else
659 chanban_add (c, userv[i], bantype);
660 }
661 else
662 {
663 userc = sjtoken (parv[j++], ',', userv);
664
665 prefix[0] = '\0';
666 for (i = 0; i < userc; i++)
667 {
668 p = strchr (userv[i], ':');
669 if (p != NULL)
670 {
671 *p = '\0';
672 prefix[0] = '\0';
673 prefix[1] = '\0';
674 prefix[2] = '\0';
675 p++;
676 while (*p)
677 {
678 if (*p == 'o')
679 prefix[prefix[0] ? 1 : 0] = '@';
680 else if (*p == 'h')
681 prefix[prefix[0] ? 1 : 0] = '%';
682 else if (*p == 'v')
683 prefix[prefix[0] ? 1 : 0] = '+';
684 p++;
685 }
686 }
687 strlcpy (newnick, prefix, sizeof newnick);
688 strlcat (newnick, userv[i], sizeof newnick);
689 chanuser_add (c, newnick);
690 }
691 }
692 }
693
694 if (c->nummembers == 0 && !(c->modes & ircd->perm_mode))
695 channel_delete (c);
696 }
697
698 static void
699 m_part (sourceinfo_t *si, int parc, char *parv[])
700 {
701 int chanc;
702 char *chanv[256];
703 int i;
704
705 chanc = sjtoken (parv[0], ',', chanv);
706 for (i = 0; i < chanc; i++)
707 {
708 slog (LG_DEBUG, "m_part(): user left channel: %s -> %s", si->su->nick, chanv[i]);
709
710 chanuser_delete (channel_find (chanv[i]), si->su);
711 }
712 }
713
714 static void
715 m_nick (sourceinfo_t *si, int parc, char *parv[])
716 {
717 user_t *u;
718 struct in_addr ip;
719 char ipstring[64];
720 char *p;
721 int i;
722
723 /* got the right number of args for an introduction? */
724 if (parc >= 8)
725 {
726 /* -> AB N jilles 1 1137687480 jilles jaguar.test +oiwgrx jilles B]AAAB ABAAE :Jilles Tjoelker */
727 /* -> AB N test4 1 1137690148 jilles jaguar.test +iw B]AAAB ABAAG :Jilles Tjoelker */
728 slog (LG_DEBUG, "m_nick(): new user on `%s': %s", si->s->name, parv[0]);
729
730 ipstring[0] = '\0';
731 if (strlen (parv[parc - 3]) == 6)
732 {
733 ip.s_addr = ntohl (base64touint (parv[parc - 3]));
734 if (!inet_ntop (AF_INET, &ip, ipstring, sizeof ipstring))
735 ipstring[0] = '\0';
736 }
737 u = user_add (parv[0], parv[3], parv[4], NULL, ipstring, parv[parc - 2], parv[parc - 1], si->s, atoi (parv[2]));
738
739 if (parv[5][0] == '+')
740 {
741 user_mode (u, parv[5]);
742 i = 1;
743 if (strchr (parv[5], 'r'))
744 {
745 handle_burstlogin (u, parv[5 + i]);
746 /* killed to force logout? */
747 if (user_find (parv[parc - 2]) == NULL)
748 return;
749 i++;
750 }
751 if (strchr (parv[5], 'h'))
752 {
753 p = strchr (parv[5 + i], '@');
754 if (p == NULL)
755 strlcpy (u->vhost, parv[5 + i], sizeof u->vhost);
756 else
757 {
758 strlcpy (u->vhost, p + 1, sizeof u->vhost);
759 strlcpy (u->user, parv[5 + i], sizeof u->user);
760 p = strchr (u->user, '@');
761 if (p != NULL)
762 *p = '\0';
763 }
764 i++;
765 }
766 if (strchr (parv[5], 'x'))
767 {
768 u->flags |= UF_HIDEHOSTREQ;
769 /* this must be after setting the account name */
770 check_hidehost (u);
771 }
772 }
773
774 handle_nickchange (u);
775 }
776 /* if it's only 2 then it's a nickname change */
777 else if (parc == 2)
778 {
779 if (!si->su)
780 {
781 slog (LG_DEBUG, "m_nick(): server trying to change nick: %s", si->s != NULL ? si->s->name : "<none>");
782 return;
783 }
784
785 slog (LG_DEBUG, "m_nick(): nickname change from `%s': %s", si->su->nick, parv[0]);
786
787 user_changenick (si->su, parv[0], atoi (parv[1]));
788
789 handle_nickchange (si->su);
790 }
791 else
792 {
793 slog (LG_DEBUG, "m_nick(): got NICK with wrong (%d) number of params", parc);
794
795 for (i = 0; i < parc; i++)
796 slog (LG_DEBUG, "m_nick(): parv[%d] = %s", i, parv[i]);
797 }
798 }
799
800 static void
801 m_quit (sourceinfo_t *si, int parc, char *parv[])
802 {
803 slog (LG_DEBUG, "m_quit(): user leaving: %s", si->su->nick);
804
805 /* user_delete() takes care of removing channels and so forth */
806 user_delete (si->su);
807 }
808
809 static void
810 m_mode (sourceinfo_t *si, int parc, char *parv[])
811 {
812 user_t *u;
813 char *p;
814
815 if (*parv[0] == '#')
816 channel_mode (NULL, channel_find (parv[0]), parc - 1, &parv[1]);
817 else
818 {
819 /* Yes this is a nick and not a UID -- jilles */
820 u = user_find_named (parv[0]);
821 if (u == NULL)
822 {
823 slog (LG_DEBUG, "m_mode(): user mode for unknown user %s", parv[0]);
824 return;
825 }
826 user_mode (u, parv[1]);
827 if (strchr (parv[1], 'x'))
828 {
829 u->flags |= UF_HIDEHOSTREQ;
830 check_hidehost (u);
831 }
832 if (strchr (parv[1], 'h'))
833 {
834 if (parc > 2)
835 {
836 /* assume +h */
837 p = strchr (parv[2], '@');
838 if (p == NULL)
839 strlcpy (u->vhost, parv[2], sizeof u->vhost);
840 else
841 {
842 strlcpy (u->vhost, p + 1, sizeof u->vhost);
843 strlcpy (u->user, parv[2], sizeof u->user);
844 p = strchr (u->user, '@');
845 if (p != NULL)
846 *p = '\0';
847 }
848 slog (LG_DEBUG, "m_mode(): user %s setting vhost %s@%s", u->nick, u->user, u->vhost);
849 }
850 else
851 {
852 /* must be -h */
853 /* XXX we don't know the original ident */
854 slog (LG_DEBUG, "m_mode(): user %s turning off vhost", u->nick);
855 strlcpy (u->vhost, u->host, sizeof u->vhost);
856 /* revert to +x vhost if applicable */
857 check_hidehost (u);
858 }
859 }
860 }
861 }
862
863 static void
864 m_clearmode (sourceinfo_t *si, int parc, char *parv[])
865 {
866 channel_t *chan;
867 char *p, c;
868 node_t *n, *tn;
869 chanuser_t *cu;
870 int i;
871
872 /* -> ABAAA CM # b */
873 /* Note: this is an IRCop command, do not enforce mode locks. */
874 chan = channel_find (parv[0]);
875 if (chan == NULL)
876 {
877 slog (LG_DEBUG, "m_clearmode(): unknown channel %s", parv[0]);
878 return;
879 }
880 p = parv[1];
881 while ((c = *p++))
882 {
883 if (c == 'b')
884 {
885 LIST_FOREACH_SAFE (n, tn, chan->bans.head)
886 {
887 if (((chanban_t *) n->data)->type == 'b')
888 chanban_delete (static_cast<chanban_t *> (n->data));
889 }
890 }
891 else if (c == 'e')
892 {
893 LIST_FOREACH_SAFE (n, tn, chan->bans.head)
894 {
895 if (((chanban_t *) n->data)->type == 'e')
896 chanban_delete (static_cast<chanban_t *> (n->data));
897 }
898 }
899 else if (c == 'k')
900 {
901 if (chan->key)
902 sfree (chan->key);
903 chan->key = NULL;
904 }
905 else if (c == 'l')
906 chan->limit = 0;
907 else if (c == 'o')
908 {
909 LIST_FOREACH (n, chan->members.head)
910 {
911 cu = (chanuser_t *) n->data;
912 if (cu->user->server == me.me)
913 {
914 /* it's a service, reop */
915 sts ("%s M %s +o %s", me.numeric, chan->name, cu->user->uid);
916 }
917 else
918 cu->modes &= ~CMODE_OP;
919 }
920 }
921 else if (c == 'h')
922 {
923 LIST_FOREACH (n, chan->members.head)
924 {
925 cu = (chanuser_t *) n->data;
926 cu->modes &= ~CMODE_HALFOP;
927 }
928 }
929 else if (c == 'v')
930 {
931 LIST_FOREACH (n, chan->members.head)
932 {
933 cu = (chanuser_t *) n->data;
934 cu->modes &= ~CMODE_VOICE;
935 }
936 }
937 else
938 for (i = 0; mode_list[i].mode != '\0'; i++)
939 {
940 if (c == mode_list[i].mode)
941 chan->modes &= ~mode_list[i].value;
942 }
943 }
944 }
945
946 static void
947 m_kick (sourceinfo_t *si, int parc, char *parv[])
948 {
949 user_t *u = user_find (parv[1]);
950 channel_t *c = channel_find (parv[0]);
951
952 /* -> :rakaur KICK #shrike rintaun :test */
953 slog (LG_DEBUG, "m_kick(): user was kicked: %s -> %s", parv[1], parv[0]);
954
955 if (!u)
956 {
957 slog (LG_DEBUG, "m_kick(): got kick for nonexistant user %s", parv[1]);
958 return;
959 }
960
961 if (!c)
962 {
963 slog (LG_DEBUG, "m_kick(): got kick in nonexistant channel: %s", parv[0]);
964 return;
965 }
966
967 if (!chanuser_find (c, u))
968 {
969 slog (LG_DEBUG, "m_kick(): got kick for %s not in %s", u->nick, c->name);
970 return;
971 }
972
973 chanuser_delete (c, u);
974
975 /* if they kicked us, let's rejoin */
976 if (is_internal_client (u))
977 {
978 slog (LG_DEBUG, "m_kick(): %s got kicked from %s; rejoining", u->nick, parv[0]);
979 join (parv[0], u->nick);
980 }
981 }
982
983 static void
984 m_kill (sourceinfo_t *si, int parc, char *parv[])
985 {
986 handle_kill (si, parv[0], parc > 1 ? parv[1] : "<No reason given>");
987 }
988
989 static void
990 m_squit (sourceinfo_t *si, int parc, char *parv[])
991 {
992 slog (LG_DEBUG, "m_squit(): server leaving: %s from %s", parv[0], parv[1]);
993 server_delete (parv[0]);
994 }
995
996 /* SERVER ircu.devel.atheme.org 1 1119902586 1119908830 J10 ABAP] + :lets lol */
997 static void
998 m_server (sourceinfo_t *si, int parc, char *parv[])
999 {
1000 server_t *s;
1001
1002 /* We dont care about the max connections. */
1003 parv[5][2] = '\0';
1004
1005 slog (LG_DEBUG, "m_server(): new server: %s, id %s, %s", parv[0], parv[5], parv[4][0] == 'P' ? "eob" : "bursting");
1006 s = handle_server (si, parv[0], parv[5], atoi (parv[1]), parv[7]);
1007
1008 /* SF_EOB may only be set when we have all users on the server.
1009 * so store the fact that they are EOB in another flag.
1010 * handle_eob() will set SF_EOB when the uplink has finished bursting.
1011 * -- jilles */
1012 if (s != NULL && parv[4][0] == 'P')
1013 s->flags |= SF_EOB2;
1014 }
1015
1016 static void
1017 m_stats (sourceinfo_t *si, int parc, char *parv[])
1018 {
1019 handle_stats (si->su, parv[0][0]);
1020 }
1021
1022 static void
1023 m_admin (sourceinfo_t *si, int parc, char *parv[])
1024 {
1025 handle_admin (si->su);
1026 }
1027
1028 static void
1029 m_version (sourceinfo_t *si, int parc, char *parv[])
1030 {
1031 handle_version (si->su);
1032 }
1033
1034 static void
1035 m_info (sourceinfo_t *si, int parc, char *parv[])
1036 {
1037 handle_info (si->su);
1038 }
1039
1040 static void
1041 m_motd (sourceinfo_t *si, int parc, char *parv[])
1042 {
1043 handle_motd (si->su);
1044 }
1045
1046 static void
1047 m_whois (sourceinfo_t *si, int parc, char *parv[])
1048 {
1049 handle_whois (si->su, parv[1]);
1050 }
1051
1052 static void
1053 m_trace (sourceinfo_t *si, int parc, char *parv[])
1054 {
1055 handle_trace (si->su, parv[0], parc >= 2 ? parv[1] : NULL);
1056 }
1057
1058 static void
1059 m_away (sourceinfo_t *si, int parc, char *parv[])
1060 {
1061 handle_away (si->su, parc >= 1 ? parv[0] : NULL);
1062 }
1063
1064 static void
1065 m_pass (sourceinfo_t *si, int parc, char *parv[])
1066 {
1067 if (strcmp (curr_uplink->pass, parv[0]))
1068 {
1069 slog (LG_INFO, "m_pass(): password mismatch from uplink; aborting");
1070 runflags |= RF_SHUTDOWN;
1071 }
1072 }
1073
1074 static void
1075 m_error (sourceinfo_t *si, int parc, char *parv[])
1076 {
1077 slog (LG_INFO, "m_error(): error from server: %s", parv[0]);
1078 }
1079
1080 static void
1081 m_eos (sourceinfo_t *si, int parc, char *parv[])
1082 {
1083 handle_eob (si->s);
1084
1085 /* acknowledge a local END_OF_BURST */
1086 if (si->s->uplink == me.me)
1087 sts ("%s EA", me.numeric);
1088 }
1089
1090 static void
1091 check_hidehost (user_t *u)
1092 {
1093 static bool warned = false;
1094
1095 /* do they qualify? */
1096 if (!(u->flags & UF_HIDEHOSTREQ) || u->myuser == NULL || (u->myuser->flags & MU_WAITAUTH))
1097 return;
1098 /* don't use this if they have some other kind of vhost */
1099 if (strcmp (u->host, u->vhost))
1100 {
1101 slog (LG_DEBUG, "check_hidehost(): +x overruled by other vhost for %s", u->nick);
1102 return;
1103 }
1104 if (me.hidehostsuffix == NULL)
1105 {
1106 if (!warned)
1107 {
1108 wallops ("Misconfiguration: serverinfo::hidehostsuffix not set");
1109 warned = true;
1110 }
1111 return;
1112 }
1113 snprintf (u->vhost, sizeof u->vhost, "%s.%s", u->myuser->name, me.hidehostsuffix);
1114 slog (LG_DEBUG, "check_hidehost(): %s -> %s", u->nick, u->vhost);
1115 }
1116
1117 static pcommand_t const pcommands[] = {
1118 { "G", m_ping, 1, MSRC_USER | MSRC_SERVER },
1119 { "Z", m_pong, 1, MSRC_SERVER },
1120 { "P", m_privmsg, 2, MSRC_USER },
1121 { "O", m_notice, 2, MSRC_USER | MSRC_SERVER },
1122 { "NOTICE", m_notice, 2, MSRC_UNREG },
1123 { "C", m_create, 1, MSRC_USER },
1124 { "J", m_join, 1, MSRC_USER },
1125 { "EB", m_eos, 0, MSRC_SERVER },
1126 { "B", m_burst, 2, MSRC_SERVER },
1127 { "L", m_part, 1, MSRC_USER },
1128 { "N", m_nick, 2, MSRC_USER | MSRC_SERVER },
1129 { "Q", m_quit, 1, MSRC_USER },
1130 { "M", m_mode, 2, MSRC_USER | MSRC_SERVER },
1131 { "OM", m_mode, 2, MSRC_USER }, /* OPMODE, treat as MODE */
1132 { "CM", m_clearmode, 2, MSRC_USER },
1133 { "K", m_kick, 2, MSRC_USER | MSRC_SERVER },
1134 { "D", m_kill, 1, MSRC_USER | MSRC_SERVER },
1135 { "SQ", m_squit, 1, MSRC_USER | MSRC_SERVER },
1136 { "S", m_server, 8, MSRC_SERVER },
1137 { "SERVER", m_server, 8, MSRC_UNREG },
1138 { "R", m_stats, 2, MSRC_USER },
1139 { "AD", m_admin, 1, MSRC_USER },
1140 { "V", m_version, 1, MSRC_USER },
1141 { "F", m_info, 1, MSRC_USER },
1142 { "W", m_whois, 2, MSRC_USER },
1143 { "TR", m_trace, 1, MSRC_USER },
1144 { "A", m_away, 0, MSRC_USER },
1145 { "PASS", m_pass, 1, MSRC_UNREG },
1146 { "Y", m_error, 1, MSRC_UNREG | MSRC_SERVER },
1147 { "ERROR", m_error, 1, MSRC_UNREG | MSRC_SERVER },
1148 { "T", m_topic, 2, MSRC_USER | MSRC_SERVER },
1149 { "MO", m_motd, 1, MSRC_USER },
1150 };
1151
1152 nefarious_handler::nefarious_handler ()
1153 {
1154 parse = &p10_parse;
1155
1156 mode_list = nefarious_mode_list;
1157 ignore_mode_list = nefarious_ignore_mode_list;
1158 status_mode_list = nefarious_status_mode_list;
1159 prefix_mode_list = nefarious_prefix_mode_list;
1160
1161 ircd = &Nefarious;
1162
1163 pcommand_add (&pcommands);
1164 }
1165
1166 nefarious_handler::~nefarious_handler ()
1167 {
1168 parse = NULL;
1169
1170 mode_list = NULL;
1171 ignore_mode_list = NULL;
1172 status_mode_list = NULL;
1173 prefix_mode_list = NULL;
1174
1175 ircd = NULL;
1176
1177 pcommand_delete (&pcommands);
1178 }
1179 } // namespace protocol
1180
1181 #define FACREG_TYPE protocol::nefarious_handler
1182 #define FACREG_TYPE_NAME "nefarious"
1183 #define FACREG_INTERFACE_TYPE protocol::handler
1184 #include <ermyth/factory_reg.h>