]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/dnsqueue.cpp
Fixing sendq crask
[user/henk/code/inspircd.git] / src / dnsqueue.cpp
1 /*       +------------------------------------+
2  *       | Inspire Internet Relay Chat Daemon |
3  *       +------------------------------------+
4  *
5  *  InspIRCd is copyright (C) 2002-2006 ChatSpike-Dev.
6  *                       E-mail:
7  *                <brain@chatspike.net>
8  *                <Craig@chatspike.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 #include "inspircd_config.h"
18 #include "inspircd.h"
19 #include "configreader.h"
20 #include <unistd.h>
21 #include <sys/errno.h>
22 #include <sys/ioctl.h>
23 #include <sys/utsname.h>
24 #include <string>
25 #include "users.h"
26 #include "globals.h"
27 #include "inspstring.h"
28 #include "dnsqueue.h"
29 #include "dns.h"
30 #include "helperfuncs.h"
31 #include "hashcomp.h"
32 #include "socketengine.h"
33 #include "socket.h"
34
35 extern ServerConfig* Config;
36 extern InspIRCd* ServerInstance;
37
38 class Lookup;
39
40 Lookup* dnslist[MAX_DESCRIPTORS];
41 Lookup* user_fd_to_dns[MAX_DESCRIPTORS];
42
43 //enum LookupState { reverse, forward };
44
45 class Lookup {
46 private:
47         char u[NICKMAX];
48         std::string hostname;
49 public:
50         DNS resolver1;
51         DNS resolver2;
52
53         Lookup()
54         {
55                 *u = 0;
56                 hostname = "";
57         }
58
59         void Reset()
60         {
61                 *u = 0;
62                 hostname = "";
63         }
64
65         ~Lookup()
66         {
67         }
68
69         bool DoLookup(std::string nick)
70         {
71                 hostname = "";
72                 userrec* usr = Find(nick);
73                 if (usr)
74                 {
75                         resolver1.SetNS(std::string(Config->DNSServer));
76                         if (!resolver1.ReverseLookup(std::string(usr->host), true))
77                         {
78                                 return false;
79                         }
80                         strlcpy(u,nick.c_str(),NICKMAX-1);
81
82                         /* ASSOCIATE WITH DNS LOOKUP LIST */
83                         if (resolver1.GetFD() != -1)
84                         {
85                                 dnslist[resolver1.GetFD()] = this;
86                                 user_fd_to_dns[usr->fd] = this;
87                                 return true;
88                         }
89                 }
90                 return false;
91         }
92
93         bool Done(int fdcheck)
94         {
95                 if (hostname != "")
96                 {
97                         // doing forward lookup
98                         userrec* usr = NULL;
99                         if (resolver2.HasResult(fdcheck))
100                         {
101                                 if (resolver2.GetFD() != -1)
102                                 {
103                                         dnslist[resolver2.GetFD()] = NULL;
104                                         std::string ip = resolver2.GetResultIP();
105                                         usr = Find(u);
106                                         if (usr)
107                                         {
108                                                 if (usr->registered > 3)
109                                                 {
110                                                         usr->dns_done = true;
111                                                         return true;
112                                                 }
113                                                 if ((hostname != "") && (usr->registered != 7))
114                                                 {
115                                                         if ((std::string((char*)inet_ntoa(usr->ip4)) == ip) && (hostname.length() < 65))
116                                                         {
117                                                                 strlcpy(usr->host,hostname.c_str(),64);
118                                                                 strlcpy(usr->dhost,hostname.c_str(),64);
119                                                                 /*address_cache::iterator address = addrcache.find(usr->ip4);
120                                                                 if (address == addrcache.end())
121                                                                 {
122                                                                         log(DEBUG,"Caching hostname %s -> %s",(char*)inet_ntoa(usr->ip4),hostname.c_str());
123                                                                         addrcache[usr->ip4] = new std::string(hostname);
124                                                                 }*/
125                                                                 WriteServ(usr->fd,"NOTICE Auth :*** Found your hostname");
126                                                         }
127                                                         usr->dns_done = true;
128                                                         return true;
129                                                 }
130                                         }
131                                 }
132                                 else
133                                 {
134                                         usr = Find(u);
135                                         if (usr)
136                                         {
137                                                 usr->dns_done = true;
138                                         }
139                                         return true;
140                                 }
141                         }
142                         return false;
143                 }
144                 else
145                 {
146                         // doing reverse lookup
147                         userrec* usr = NULL;
148                         if (resolver1.HasResult(fdcheck))
149                         {
150                                 usr = Find(u);
151                                 if ((usr) && (usr->dns_done))
152                                 {
153                                         if (resolver1.GetFD() != -1)
154                                         {
155                                                 dnslist[resolver1.GetFD()] = NULL;
156                                                 user_fd_to_dns[usr->fd] = NULL;
157                                         }
158                                         return true;
159                                 }
160                                 if (resolver1.GetFD() != -1)
161                                 {
162                                         dnslist[resolver1.GetFD()] = NULL;
163                                         hostname = resolver1.GetResult();
164                                         if (usr)
165                                         {
166                                                 user_fd_to_dns[usr->fd] = NULL;
167                                                 if ((usr->registered > 3) || (hostname == ""))
168                                                 {
169                                                         WriteServ(usr->fd,"NOTICE Auth :*** Could not resolve your hostname -- Using your IP address instead");
170                                                         usr->dns_done = true;
171                                                         return true;
172                                                 }
173                                         }
174                                         if (hostname != "")
175                                         {
176                                                 resolver2.ForwardLookup(hostname, true);
177                                                 if (resolver2.GetFD() != -1)
178                                                 {
179                                                         dnslist[resolver2.GetFD()] = this;
180                                                         if (usr)
181                                                                 user_fd_to_dns[usr->fd] = this;
182                                                 }
183                                         }
184                                 }
185                         }
186                 }
187                 return false;
188         }
189
190         int GetFD()
191         {
192                 userrec* usr = Find(u);
193                 if (!usr)
194                         return 0;
195                 if (usr->dns_done)
196                         return 0;
197                 return usr->fd;
198         }
199 };
200
201 bool lookup_dns(const std::string &nick)
202 {
203         /* First attempt to find the nickname */
204         userrec* u = Find(nick);
205         if (u)
206         {
207                 /* Check the cache */
208                 /*address_cache::iterator address = addrcache.find(u->ip4);
209                 if (address != addrcache.end())
210                 {
211                         WriteServ(u->fd,"NOTICE Auth :*** Found your hostname (cached)");
212                         log(DEBUG,"Found cached host");
213                         strlcpy(u->host,address->second->c_str(),MAXBUF);
214                         strlcpy(u->dhost,address->second->c_str(),MAXBUF);
215                         u->dns_done = true;
216                         return true;
217                 }*/
218                 /* If the user exists, create a new
219                  * lookup object, and associate it
220                  * with the user. The lookup object
221                  * will maintain the reference table
222                  * which we use for quickly finding
223                  * dns results. Please note that we
224                  * do not associate a lookup with a
225                  * userrec* pointer and we use the
226                  * nickname instead because, by the
227                  * time the DNS lookup has completed,
228                  * the nickname could have quit and
229                  * if we then try and access the
230                  * pointer we get a nice segfault.
231                  */
232                 Lookup* L = new Lookup();
233                 L->DoLookup(nick);
234                 return true;
235         }
236         return false;
237 }
238
239 void ZapThisDns(int fd)
240 {
241         if ((fd < 0) || (fd > MAX_DESCRIPTORS))
242                 return;
243
244         Lookup *x = user_fd_to_dns[fd];
245
246         if (x)
247         {
248                 if (x->resolver1.GetFD() > 0)
249                 {
250                         log(DEBUG,"Whacked resolver1");
251                         dns_close(x->resolver1.GetFD());
252                 }
253
254                 if (x->resolver2.GetFD() > 0)
255                 {
256                         log(DEBUG,"Whacked resolver2");
257                         dns_close(x->resolver2.GetFD());
258                 }
259         }
260 }
261
262 void dns_poll(int fdcheck)
263 {
264         /* Check the given file descriptor is in valid range */
265         if ((fdcheck < 0) || (fdcheck > MAX_DESCRIPTORS))
266                 return;
267
268         /* Try and find the file descriptor in our list of
269          * active DNS lookups
270          */
271         Lookup *x = dnslist[fdcheck];
272         if (x)
273         {
274                 /* If it exists check if its a valid fd still */
275                 if (x->GetFD() != -1)
276                 {
277                         /* Check if its done, if it is delete it */
278                         if (x->Done(fdcheck))
279                         {
280                                 /* We don't need to delete the file descriptor
281                                  * from the socket engine, as dns.cpp tracks it
282                                  * for us if we are in single-threaded country.
283                                  */
284                                 DELETE(x);
285                         }
286                 }
287                 else
288                 {
289                         /* its fd is dodgy, the dns code probably
290                          * bashed it due to error. Free the class.
291                          */
292                         DELETE(x);
293                 }
294                 /* If we got down here, the dns lookup was valid, BUT,
295                  * its still in progress. Be patient, and wait for
296                  * more socketengine events to complete the lookups.
297                  */
298                 return;
299         }
300         /* This FD doesnt belong here, lets be rid of it,
301          * just to be safe so we dont get any more events
302          * about it.
303          */
304         if (ServerInstance && ServerInstance->SE)
305                 ServerInstance->SE->DelFd(fdcheck);
306 }