]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/cmd_nick.cpp
Fixed spanningtree to not allow malformed conf tags (missing/empty values)
[user/henk/code/inspircd.git] / src / cmd_nick.cpp
1 /*       +------------------------------------+
2  *       | Inspire Internet Relay Chat Daemon |
3  *       +------------------------------------+
4  *
5  *  Inspire is copyright (C) 2002-2005 ChatSpike-Dev.
6  *                       E-mail:
7  *                <brain.net>
8  *                <Craig.net>
9  *
10  * Written by Craig Edwards, Craig McLure, and others.
11  * This program is free but copyrighted software; see
12  *            the file COPYING for details.
13  *
14  * ---------------------------------------------------
15  */
16
17 using namespace std;
18
19 #include "inspircd_config.h"
20 #include "inspircd.h"
21 #include "inspircd_io.h"
22 #include <time.h>
23 #include <string>
24 #ifdef GCC3
25 #include <ext/hash_map>
26 #else
27 #include <hash_map>
28 #endif
29 #include <map>
30 #include <sstream>
31 #include <vector>
32 #include <deque>
33 #include "users.h"
34 #include "ctables.h"
35 #include "globals.h"
36 #include "modules.h"
37 #include "dynamic.h"
38 #include "wildcard.h"
39 #include "message.h"
40 #include "commands.h"
41 #include "mode.h"
42 #include "xline.h"
43 #include "inspstring.h"
44 #include "dnsqueue.h"
45 #include "dns.h"
46 #include "helperfuncs.h"
47 #include "hashcomp.h"
48 #include "socketengine.h"
49 #include "typedefs.h"
50 #include "command_parse.h"
51 #include "cmd_nick.h"
52
53 extern ServerConfig* Config;
54 extern InspIRCd* ServerInstance;
55 extern int MODCOUNT;
56 extern std::vector<Module*> modules;
57 extern std::vector<ircd_module*> factory;
58 extern time_t TIME;
59 extern user_hash clientlist;
60 extern chan_hash chanlist;
61 extern whowas_hash whowas;
62 extern std::vector<userrec*> all_opers;
63 extern std::vector<userrec*> local_users;
64 extern userrec* fd_ref_table[65536];
65
66 void cmd_nick::Handle (char **parameters, int pcnt, userrec *user)
67 {
68         if (pcnt < 1) 
69         {
70                 log(DEBUG,"not enough params for handle_nick");
71                 return;
72         }
73         if (!parameters[0])
74         {
75                 log(DEBUG,"invalid parameter passed to handle_nick");
76                 return;
77         }
78         if (!parameters[0][0])
79         {
80                 log(DEBUG,"zero length new nick passed to handle_nick");
81                 return;
82         }
83         if (!user)
84         {
85                 log(DEBUG,"invalid user passed to handle_nick");
86                 return;
87         }
88         if (!user->nick)
89         {
90                 log(DEBUG,"invalid old nick passed to handle_nick");
91                 return;
92         }
93         if (!strcasecmp(user->nick,parameters[0]))
94         {
95                 log(DEBUG,"old nick is new nick, skipping");
96                 return;
97         }
98         else
99         {
100                 if (strlen(parameters[0]) > 1)
101                 {
102                         if (parameters[0][0] == ':')
103                         {
104                                 *parameters[0]++;
105                         }
106                 }
107                 if (matches_qline(parameters[0]))
108                 {
109                         WriteOpers("*** Q-Lined nickname %s from %s!%s@%s: %s",parameters[0],user->nick,user->ident,user->host,matches_qline(parameters[0]));
110                         WriteServ(user->fd,"432 %s %s :Invalid nickname: %s",user->nick,parameters[0],matches_qline(parameters[0]));
111                         return;
112                 }
113                 if ((Find(parameters[0])) && (Find(parameters[0]) != user))
114                 {
115                         WriteServ(user->fd,"433 %s %s :Nickname is already in use.",user->nick,parameters[0]);
116                         return;
117                 }
118         }
119         if (isnick(parameters[0]) == 0)
120         {
121                 WriteServ(user->fd,"432 %s %s :Erroneous Nickname",user->nick,parameters[0]);
122                 return;
123         }
124
125         if (user->registered == 7)
126         {
127                 int MOD_RESULT = 0;
128                 FOREACH_RESULT(OnUserPreNick(user,parameters[0]));
129                 if (MOD_RESULT) {
130                         // if a module returns true, the nick change is silently forbidden.
131                         return;
132                 }
133
134                 WriteCommon(user,"NICK %s",parameters[0]);
135                 
136         }
137         
138         char oldnick[NICKMAX];
139         strlcpy(oldnick,user->nick,NICKMAX);
140
141         /* change the nick of the user in the users_hash */
142         user = ReHashNick(user->nick, parameters[0]);
143         /* actually change the nick within the record */
144         if (!user) return;
145         if (!user->nick) return;
146
147         strlcpy(user->nick, parameters[0],NICKMAX);
148
149         log(DEBUG,"new nick set: %s",user->nick);
150         
151         if (user->registered < 3)
152         {
153                 user->registered = (user->registered | 2);
154                 // dont attempt to look up the dns until they pick a nick... because otherwise their pointer WILL change
155                 // and unless we're lucky we'll get a duff one later on.
156                 //user->dns_done = (!lookup_dns(user->nick));
157                 //if (user->dns_done)
158                 //      log(DEBUG,"Aborting dns lookup of %s because dns server experienced a failure.",user->nick);
159
160 #ifdef THREADED_DNS
161                 // initialize their dns lookup thread
162                 if (pthread_create(&user->dnsthread, NULL, dns_task, (void *)user) != 0)
163                 {
164                         log(DEBUG,"Failed to create DNS lookup thread for user %s",user->nick);
165                 }
166 #else
167                 user->dns_done = (!lookup_dns(user->nick));
168                 if (user->dns_done)
169                         log(DEBUG,"Aborting dns lookup of %s because dns server experienced a failure.",user->nick);
170 #endif
171         
172         }
173         if (user->registered == 3)
174         {
175                 /* user is registered now, bit 0 = USER command, bit 1 = sent a NICK command */
176                 FOREACH_MOD OnUserRegister(user);
177                 ConnectUser(user);
178         }
179         if (user->registered == 7)
180         {
181                 FOREACH_MOD OnUserPostNick(user,oldnick);
182         }
183 }
184