ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/trunk/src/parse.c
Revision: 3156
Committed: Fri Mar 14 19:57:38 2014 UTC (11 years, 5 months ago) by michael
Content type: text/x-csrc
File size: 21935 byte(s)
Log Message:
- Removed client_p pointers from everywhere

File Contents

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

Properties

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