ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/trunk/src/parse.c
Revision: 6437
Committed: Thu Aug 27 13:08:57 2015 UTC (10 years ago) by michael
Content type: text/x-csrc
File size: 18787 byte(s)
Log Message:
- parse.c:parse_remove_unknown(): use get_client_name()

File Contents

# Content
1 /*
2 * ircd-hybrid: an advanced, lightweight Internet Relay Chat Daemon (ircd)
3 *
4 * Copyright (c) 1997-2015 ircd-hybrid development team
5 *
6 * This program 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 this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
19 * USA
20 */
21
22 /*! \file parse.c
23 * \brief The message parser.
24 * \version $Id$
25 */
26
27 #include "stdinc.h"
28 #include "client.h"
29 #include "parse.h"
30 #include "channel.h"
31 #include "hash.h"
32 #include "id.h"
33 #include "irc_string.h"
34 #include "ircd.h"
35 #include "numeric.h"
36 #include "log.h"
37 #include "send.h"
38 #include "conf.h"
39 #include "memory.h"
40 #include "user.h"
41 #include "server.h"
42
43
44 /*
45 * (based on orabidoo's parser code)
46 *
47 * This has always just been a trie. Look at volume III of Knuth ACP
48 *
49 *
50 * ok, you start out with an array of pointers, each one corresponds
51 * to a letter at the current position in the command being examined.
52 *
53 * so roughly you have this for matching 'trie' or 'tie'
54 *
55 * 't' points -> [MessageTree *] 'r' -> [MessageTree *] -> 'i'
56 * -> [MessageTree *] -> [MessageTree *] -> 'e' and matches
57 *
58 * 'i' -> [MessageTree *] -> 'e' and matches
59 *
60 * BUGS (Limitations!)
61 *
62 * I designed this trie to parse ircd commands. Hence it currently
63 * casefolds. This is trivial to fix by increasing MAXPTRLEN.
64 * This trie also "folds" '{' etc. down. This means, the input to this
65 * trie must be alpha tokens only. This again, is a limitation that
66 * can be overcome by increasing MAXPTRLEN to include upper/lower case
67 * at the expense of more memory. At the extreme end, you could make
68 * MAXPTRLEN 128.
69 *
70 * This is also not a patricia trie. On short ircd tokens, this is
71 * not likely going to matter.
72 *
73 * Diane Bruce (Dianora), June 6 2003
74 */
75
76 /*
77 * Must be a power of 2, and larger than 26 [a-z]|[A-Z]. It's used to allocate
78 * the set of pointers at each node of the tree.
79 * There are MAXPTRLEN pointers at each node. Obviously, there have to be more
80 * pointers than ASCII letters. 32 is a nice number since there is then no need
81 * to shift 'A'/'a' to base 0 index, at the expense of a few never used
82 * pointers.
83 * For a small parser like this, this is a good compromise and does
84 * make it somewhat faster. - Dianora
85 */
86 enum { MAXPTRLEN = 32 };
87
88
89 static struct MessageTree
90 {
91 int links; /* Count of all pointers (including msg) at this node
92 * used as reference count for deletion of _this_ node.
93 */
94 struct Message *msg;
95 struct MessageTree *pointers[MAXPTRLEN];
96 } msg_tree;
97
98
99 /* remove_unknown()
100 *
101 * inputs -
102 * output -
103 * side effects -
104 */
105 static void
106 parse_remove_unknown(struct Client *client_p, const char *lsender, char *lbuffer)
107 {
108 /*
109 * Do kill if it came from a server because it means there is a ghost
110 * user on the other server which needs to be removed. -avalon
111 * Tell opers about this. -Taner
112 */
113 /*
114 * '[0-9]something' is an ID (KILL/SQUIT depending on its length)
115 * 'nodots' is a nickname (KILL)
116 * 'no.dot.at.start' is a server (SQUIT)
117 */
118 if ((IsDigit(*lsender) && strlen(lsender) <= IRC_MAXSID) || strchr(lsender, '.'))
119 {
120 sendto_realops_flags(UMODE_DEBUG, L_ADMIN, SEND_NOTICE,
121 "Unknown prefix (%s) from %s, Squitting %s",
122 lbuffer, get_client_name(client_p, SHOW_IP), lsender);
123 sendto_realops_flags(UMODE_DEBUG, L_OPER, SEND_NOTICE,
124 "Unknown prefix (%s) from %s, Squitting %s",
125 lbuffer, get_client_name(client_p, MASK_IP), lsender);
126 sendto_one(client_p, ":%s SQUIT %s :(Unknown prefix (%s) from %s)",
127 me.id, lsender, lbuffer, client_p->name);
128 }
129 else
130 sendto_one(client_p, ":%s KILL %s :%s (Unknown Client)",
131 me.id, lsender, me.name);
132 }
133
134 /*
135 *
136 * parc number of arguments ('sender' counted as one!)
137 * parv[0] pointer to 'sender' (may point to empty string) (not used)
138 * parv[1]..parv[parc-1]
139 * pointers to additional parameters, this is a NULL
140 * terminated list (parv[parc] == NULL).
141 *
142 * *WARNING*
143 * Numerics are mostly error reports. If there is something
144 * wrong with the message, just *DROP* it! Don't even think of
145 * sending back a neat error message -- big danger of creating
146 * a ping pong error message...
147 *
148 * Rewritten by Nemesi, Jan 1999, to support numeric nicks in parv[1]
149 *
150 * Called when we get a numeric message from a remote _server_ and we are
151 * supposed to forward it somewhere. Note that we always ignore numerics sent
152 * to 'me' and simply drop the message if we can't handle with this properly:
153 * the savvy approach is NEVER generate an error in response to an... error :)
154 */
155 static void
156 parse_handle_numeric(unsigned int numeric, struct Client *source_p, int parc, char *parv[])
157 {
158 struct Client *target_p = NULL;
159 struct Channel *chptr = NULL;
160
161 /*
162 * Avoid trash, we need it to come from a server and have a target
163 */
164 if (parc < 2 || !IsServer(source_p))
165 return;
166
167 /*
168 * Who should receive this message ? Will we do something with it ?
169 * Note that we use findUser functions, so the target can't be neither
170 * a server, nor a channel (?) nor a list of targets (?) .. u2.10
171 * should never generate numeric replies to non-users anyway
172 * Ahem... it can be a channel actually, csc bots use it :\ --Nem
173 */
174 if (IsChanPrefix(*parv[1]))
175 chptr = hash_find_channel(parv[1]);
176 else
177 target_p = find_person(source_p, parv[1]);
178
179 if ((!target_p || target_p->from == source_p->from) && !chptr)
180 return;
181
182 /*
183 * Remap low number numerics, not that I understand WHY.. --Nemesi
184 */
185 /*
186 * Numerics below 100 talk about the current 'connection', you're not
187 * connected to a remote server so it doesn't make sense to send them
188 * remotely - but the information they contain may be useful, so we
189 * remap them up. Weird, but true. -- Isomer
190 */
191 if (numeric < 100)
192 numeric += 100;
193
194 if (target_p)
195 {
196 /* Fake it for server hiding, if it's our client */
197 if ((ConfigServerHide.hide_servers || IsHidden(source_p)) && MyConnect(target_p) &&
198 !HasUMode(target_p, UMODE_OPER))
199 sendto_one_numeric(target_p, &me, numeric | SND_EXPLICIT, "%s", parv[2]);
200 else
201 sendto_one_numeric(target_p, source_p, numeric | SND_EXPLICIT, "%s", parv[2]);
202 }
203 else
204 sendto_channel_butone(source_p, source_p, chptr, 0, "%u %s %s",
205 numeric, chptr->name, parv[2]);
206 }
207
208 /* handle_command()
209 *
210 * inputs - pointer to message block
211 * - pointer to client
212 * - pointer to client message is from
213 * - count of number of args
214 * - pointer to argv[] array
215 * output - -1 if error from server
216 * side effects -
217 */
218 static void
219 parse_handle_command(struct Message *message, struct Client *source_p,
220 unsigned int i, char *para[])
221 {
222 if (IsServer(source_p->from))
223 ++message->rcount;
224
225 ++message->count;
226
227 /* Check right amount of parameters is passed... --is */
228 if (i < message->args_min)
229 sendto_one_numeric(source_p, &me, ERR_NEEDMOREPARAMS, message->cmd);
230 else
231 message->handlers[source_p->from->handler](source_p, i, para);
232 }
233
234 /*
235 * parse a buffer.
236 *
237 * NOTE: parse() should not be called recusively by any other functions!
238 */
239 void
240 parse(struct Client *client_p, char *pbuffer, char *bufend)
241 {
242 struct Client *from = client_p;
243 struct Message *message = NULL;
244 char *para[MAXPARA + 2]; /* <command> + <parameters> + NULL */
245 char *ch = NULL;
246 char *s = NULL;
247 unsigned int numeric = 0;
248 unsigned int parc = 0;
249 unsigned int paramcount;
250
251 if (IsDefunct(client_p))
252 return;
253
254 assert(client_p->connection->fd.flags.open);
255 assert((bufend - pbuffer) < IRCD_BUFSIZE);
256
257 for (ch = pbuffer; *ch == ' '; ++ch) /* Skip spaces */
258 ;
259
260 if (*ch == ':')
261 {
262 /*
263 * Copy the prefix to 'sender' assuming it terminates
264 * with SPACE (or NULL, which is an error, though).
265 */
266 const char *const sender = ++ch;
267
268 if ((s = strchr(ch, ' ')))
269 {
270 *s = '\0';
271 ch = ++s;
272 }
273
274 if (*sender && IsServer(client_p))
275 {
276 if ((from = hash_find_id(sender)) == NULL)
277 from = hash_find_client(sender);
278
279 /*
280 * Hmm! If the client corresponding to the prefix is not found--what is
281 * the correct action??? Now, I will ignore the message (old IRC just
282 * let it through as if the prefix just wasn't there...) --msa
283 */
284 if (from == NULL)
285 {
286 ++ServerStats.is_unpf;
287 parse_remove_unknown(client_p, sender, pbuffer);
288 return;
289 }
290
291 if (from->from != client_p)
292 {
293 ++ServerStats.is_wrdi;
294 sendto_realops_flags(UMODE_DEBUG, L_ADMIN, SEND_NOTICE,
295 "Fake direction: dropped message from %s[%s] via %s",
296 from->name, from->from->name,
297 get_client_name(client_p, SHOW_IP));
298 sendto_realops_flags(UMODE_DEBUG, L_OPER, SEND_NOTICE,
299 "Fake direction: dropped message from %s[%s] via %s",
300 from->name, from->from->name,
301 get_client_name(client_p, MASK_IP));
302 return;
303 }
304 }
305
306 while (*ch == ' ')
307 ++ch;
308 }
309
310 if (*ch == '\0')
311 {
312 ++ServerStats.is_empt;
313 return;
314 }
315
316 /*
317 * Extract the command code from the packet. Point s to the end
318 * of the command code and calculate the length using pointer
319 * arithmetic. Note: only need length for numerics and *all*
320 * numerics must have parameters and thus a space after the command
321 * code. -avalon
322 */
323
324 /* EOB is 3 characters long but is not a numeric */
325 if (*(ch + 3) == ' ' && /* Ok, let's see if it's a possible numeric */
326 IsDigit(*ch) && IsDigit(*(ch + 1)) && IsDigit(*(ch + 2)))
327 {
328 numeric = (*ch - '0') * 100 + (*(ch + 1) - '0') * 10 + (*(ch + 2) - '0');
329 paramcount = 2; /* Destination, and the rest of it */
330 ++ServerStats.is_num;
331 s = ch + 3; /* I know this is ' ' from above if */
332 *s++ = '\0'; /* Blow away the ' ', and point s to next part */
333 }
334 else
335 {
336 unsigned int length = 0;
337
338 if ((s = strchr(ch, ' ')))
339 *s++ = '\0';
340
341 if ((message = find_command(ch)) == NULL)
342 {
343 /*
344 * Note: Give error message *only* to recognized
345 * persons. It's a nightmare situation to have
346 * two programs sending "Unknown command"'s or
347 * equivalent to each other at full blast....
348 * If it has got to person state, it at least
349 * seems to be well behaving. Perhaps this message
350 * should never be generated, though... --msa
351 * Hm, when is the buffer empty -- if a command
352 * code has been found ?? -Armin
353 */
354 if (*pbuffer)
355 if (IsClient(from))
356 sendto_one_numeric(from, &me, ERR_UNKNOWNCOMMAND, ch);
357
358 ++ServerStats.is_unco;
359 return;
360 }
361
362 assert(message->cmd);
363
364 paramcount = message->args_max;
365 length = bufend - ((s) ? s : ch);
366 message->bytes += length;
367 }
368
369 /*
370 * Must the following loop really be so devious? On surface it
371 * splits the message to parameters from blank spaces. But, if
372 * paramcount has been reached, the rest of the message goes into
373 * this last parameter (about same effect as ":" has...) --msa
374 */
375
376 /* Note initially true: s == NULL || *(s - 1) == '\0' !! */
377
378 para[parc] = ch;
379
380 if (message && (message->flags & MFLG_EXTRA))
381 {
382 /*
383 * XXX: This will have to go away after the command handler rewrite
384 */
385 para[++parc] = message->extra;
386 }
387
388 if (s)
389 {
390 if (paramcount > MAXPARA)
391 paramcount = MAXPARA;
392
393 while (1)
394 {
395 while (*s == ' ')
396 *s++ = '\0';
397
398 if (*s == '\0')
399 break;
400
401 if (*s == ':')
402 {
403 /* The rest is single parameter--can include blanks also. */
404 para[++parc] = s + (!numeric); /* Keep the colon if it's a numeric */
405 break;
406 }
407
408 para[++parc] = s;
409
410 if (parc >= paramcount)
411 break;
412
413 while (*s && *s != ' ')
414 ++s;
415 }
416 }
417
418 para[++parc] = NULL;
419
420 if (message)
421 parse_handle_command(message, from, parc, para);
422 else
423 parse_handle_numeric(numeric, from, parc, para);
424 }
425
426 /* add_msg_element()
427 *
428 * inputs - pointer to MessageTree
429 * - pointer to Message to add for given command
430 * - pointer to current portion of command being added
431 * output - NONE
432 * side effects - recursively build the Message Tree ;-)
433 */
434 /*
435 * How this works.
436 *
437 * The code first checks to see if its reached the end of the command
438 * If so, that struct MessageTree has a msg pointer updated and the links
439 * count incremented, since a msg pointer is a reference.
440 * Then the code descends recursively, building the trie.
441 * If a pointer index inside the struct MessageTree is NULL a new
442 * child struct MessageTree has to be allocated.
443 * The links (reference count) is incremented as they are created
444 * in the parent.
445 */
446 static void
447 add_msg_element(struct MessageTree *mtree_p, struct Message *msg_p,
448 const char *cmd)
449 {
450 struct MessageTree *ntree_p = NULL;
451
452 if (*cmd == '\0')
453 {
454 mtree_p->msg = msg_p;
455 mtree_p->links++; /* Have msg pointer, so up ref count */
456 }
457 else
458 {
459 /*
460 * *cmd & (MAXPTRLEN-1)
461 * convert the char pointed to at *cmd from ASCII to an integer
462 * between 0 and MAXPTRLEN.
463 * Thus 'A' -> 0x1 'B' -> 0x2 'c' -> 0x3 etc.
464 */
465 if ((ntree_p = mtree_p->pointers[*cmd & (MAXPTRLEN - 1)]) == NULL)
466 {
467 ntree_p = MyCalloc(sizeof(struct MessageTree));
468 mtree_p->pointers[*cmd & (MAXPTRLEN - 1)] = ntree_p;
469
470 mtree_p->links++; /* Have new pointer, so up ref count */
471 }
472
473 add_msg_element(ntree_p, msg_p, cmd + 1);
474 }
475 }
476
477 /* del_msg_element()
478 *
479 * inputs - Pointer to MessageTree to delete from
480 * - pointer to command name to delete
481 * output - NONE
482 * side effects - recursively deletes a token from the Message Tree ;-)
483 */
484 /*
485 * How this works.
486 *
487 * Well, first off, the code recursively descends into the trie
488 * until it finds the terminating letter of the command being removed.
489 * Once it has done that, it marks the msg pointer as NULL then
490 * reduces the reference count on that allocated struct MessageTree
491 * since a command counts as a reference.
492 *
493 * Then it pops up the recurse stack. As it comes back up the recurse
494 * The code checks to see if the child now has no pointers or msg
495 * i.e. the links count has gone to zero. If it's no longer used, the
496 * child struct MessageTree can be deleted. The parent reference
497 * to this child is then removed and the parents link count goes down.
498 * Thus, we continue to go back up removing all unused MessageTree(s)
499 */
500 static void
501 del_msg_element(struct MessageTree *mtree_p, const char *cmd)
502 {
503 struct MessageTree *ntree_p = NULL;
504
505 /*
506 * In case this is called for a nonexistent command
507 * check that there is a msg pointer here, else links-- goes -ve
508 * -db
509 */
510 if (*cmd == '\0' && mtree_p->msg)
511 {
512 mtree_p->msg = NULL;
513 mtree_p->links--;
514 }
515 else
516 {
517 if ((ntree_p = mtree_p->pointers[*cmd & (MAXPTRLEN - 1)]))
518 {
519 del_msg_element(ntree_p, cmd + 1);
520
521 if (ntree_p->links == 0)
522 {
523 mtree_p->pointers[*cmd & (MAXPTRLEN - 1)] = NULL;
524 mtree_p->links--;
525 MyFree(ntree_p);
526 }
527 }
528 }
529 }
530
531 /* msg_tree_parse()
532 *
533 * inputs - Pointer to command to find
534 * - Pointer to MessageTree root
535 * output - Find given command returning Message * if found NULL if not
536 * side effects - none
537 */
538 static struct Message *
539 msg_tree_parse(const char *cmd)
540 {
541 struct MessageTree *mtree = &msg_tree;
542
543 assert(!EmptyString(cmd));
544
545 while (IsAlpha(*cmd) && (mtree = mtree->pointers[*cmd & (MAXPTRLEN - 1)]))
546 if (*++cmd == '\0')
547 return mtree->msg;
548
549 return NULL;
550 }
551
552 /* mod_add_cmd()
553 *
554 * inputs - pointer to struct Message
555 * output - none
556 * side effects - load this one command name
557 */
558 void
559 mod_add_cmd(struct Message *msg)
560 {
561 assert(msg && msg->cmd);
562
563 /* Command already added? */
564 if (msg_tree_parse(msg->cmd))
565 return;
566
567 add_msg_element(&msg_tree, msg, msg->cmd);
568 }
569
570 /* mod_del_cmd()
571 *
572 * inputs - pointer to struct Message
573 * output - none
574 * side effects - unload this one command name
575 */
576 void
577 mod_del_cmd(struct Message *msg)
578 {
579 assert(msg && msg->cmd);
580
581 if (!msg_tree_parse(msg->cmd))
582 return;
583
584 del_msg_element(&msg_tree, msg->cmd);
585 }
586
587 /* find_command()
588 *
589 * inputs - command name
590 * output - pointer to struct Message
591 * side effects - none
592 */
593 struct Message *
594 find_command(const char *cmd)
595 {
596 return msg_tree_parse(cmd);
597 }
598
599 static void
600 recurse_report_messages(struct Client *source_p, const struct MessageTree *mtree)
601 {
602 if (mtree->msg)
603 sendto_one_numeric(source_p, &me, RPL_STATSCOMMANDS,
604 mtree->msg->cmd,
605 mtree->msg->count, mtree->msg->bytes,
606 mtree->msg->rcount);
607
608 for (unsigned int i = 0; i < MAXPTRLEN; ++i)
609 if (mtree->pointers[i])
610 recurse_report_messages(source_p, mtree->pointers[i]);
611 }
612
613 /* report_messages()
614 *
615 * inputs - pointer to client to report to
616 * output - NONE
617 * side effects - client is shown list of commands
618 */
619 void
620 report_messages(struct Client *source_p)
621 {
622 const struct MessageTree *const mtree = &msg_tree;
623
624 for (unsigned int i = 0; i < MAXPTRLEN; ++i)
625 if (mtree->pointers[i])
626 recurse_report_messages(source_p, mtree->pointers[i]);
627 }
628
629 /* m_not_oper()
630 * inputs -
631 * output -
632 * side effects - just returns a nastyogram to given user
633 */
634 int
635 m_not_oper(struct Client *source_p, int parc, char *parv[])
636 {
637 sendto_one_numeric(source_p, &me, ERR_NOPRIVILEGES);
638 return 0;
639 }
640
641 int
642 m_unregistered(struct Client *source_p, int parc, char *parv[])
643 {
644 sendto_one_numeric(source_p, &me, ERR_NOTREGISTERED);
645 return 0;
646 }
647
648 int
649 m_registered(struct Client *source_p, int parc, char *parv[])
650 {
651 sendto_one_numeric(source_p, &me, ERR_ALREADYREGISTRED);
652 return 0;
653 }
654
655 int
656 m_ignore(struct Client *source_p, int parc, char *parv[])
657 {
658 return 0;
659 }

Properties

Name Value
svn:eol-style native
svn:keywords Id Revision