]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/dnsqueue.cpp
Adjustment
[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                                                         user_fd_to_dns[usr->fd] = this;
181                                                 }
182                                         }
183                                 }
184                         }
185                 }
186                 return false;
187         }
188
189         int GetFD()
190         {
191                 userrec* usr = Find(u);
192                 if (!usr)
193                         return 0;
194                 if (usr->dns_done)
195                         return 0;
196                 return usr->fd;
197         }
198 };
199
200 bool lookup_dns(const std::string &nick)
201 {
202         /* First attempt to find the nickname */
203         userrec* u = Find(nick);
204         if (u)
205         {
206                 /* Check the cache */
207                 /*address_cache::iterator address = addrcache.find(u->ip4);
208                 if (address != addrcache.end())
209                 {
210                         WriteServ(u->fd,"NOTICE Auth :*** Found your hostname (cached)");
211                         log(DEBUG,"Found cached host");
212                         strlcpy(u->host,address->second->c_str(),MAXBUF);
213                         strlcpy(u->dhost,address->second->c_str(),MAXBUF);
214                         u->dns_done = true;
215                         return true;
216                 }*/
217                 /* If the user exists, create a new
218                  * lookup object, and associate it
219                  * with the user. The lookup object
220                  * will maintain the reference table
221                  * which we use for quickly finding
222                  * dns results. Please note that we
223                  * do not associate a lookup with a
224                  * userrec* pointer and we use the
225                  * nickname instead because, by the
226                  * time the DNS lookup has completed,
227                  * the nickname could have quit and
228                  * if we then try and access the
229                  * pointer we get a nice segfault.
230                  */
231                 Lookup* L = new Lookup();
232                 L->DoLookup(nick);
233                 return true;
234         }
235         return false;
236 }
237
238 void ZapThisDns(int fd)
239 {
240         if ((fd < 0) || (fd > MAX_DESCRIPTORS))
241                 return;
242
243         Lookup *x = user_fd_to_dns[fd];
244
245         if (x)
246         {
247                 if (x->resolver1.GetFD() > 0)
248                 {
249                         log(DEBUG,"Whacked resolver1");
250                         dns_close(x->resolver1.GetFD());
251                 }
252
253                 if (x->resolver2.GetFD() > 0)
254                 {
255                         log(DEBUG,"Whacked resolver2");
256                         dns_close(x->resolver2.GetFD());
257                 }
258         }
259 }
260
261 void dns_poll(int fdcheck)
262 {
263         /* Check the given file descriptor is in valid range */
264         if ((fdcheck < 0) || (fdcheck > MAX_DESCRIPTORS))
265                 return;
266
267         /* Try and find the file descriptor in our list of
268          * active DNS lookups
269          */
270         Lookup *x = dnslist[fdcheck];
271         if (x)
272         {
273                 /* If it exists check if its a valid fd still */
274                 if (x->GetFD() != -1)
275                 {
276                         /* Check if its done, if it is delete it */
277                         if (x->Done(fdcheck))
278                         {
279                                 /* We don't need to delete the file descriptor
280                                  * from the socket engine, as dns.cpp tracks it
281                                  * for us if we are in single-threaded country.
282                                  */
283                                 DELETE(x);
284                         }
285                 }
286                 else
287                 {
288                         /* its fd is dodgy, the dns code probably
289                          * bashed it due to error. Free the class.
290                          */
291                         DELETE(x);
292                 }
293                 /* If we got down here, the dns lookup was valid, BUT,
294                  * its still in progress. Be patient, and wait for
295                  * more socketengine events to complete the lookups.
296                  */
297                 return;
298         }
299         /* This FD doesnt belong here, lets be rid of it,
300          * just to be safe so we dont get any more events
301          * about it.
302          */
303         if (ServerInstance && ServerInstance->SE)
304                 ServerInstance->SE->DelFd(fdcheck);
305 }