ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/trunk/src/parse.c
Revision: 1726
Committed: Thu Jan 3 00:07:58 2013 UTC (12 years, 7 months ago) by michael
Content type: text/x-csrc
File size: 23528 byte(s)
Log Message:
- parse.c:handle_numeric(): add missing space as reported by Mantas

File Contents

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

Properties

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