787b336845451821833f9303eba792ae1b87ac18
[NeonServV5.git] / src / DBHelper.c
1 /* DBHelper.c - NeonServ v5.4
2  * Copyright (C) 2011-2012  Philipp Kreil (pk910)
3  * 
4  * This program is free software: you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation, either version 3 of the License, or
7  * (at your option) any later version.
8  * 
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  * 
14  * You should have received a copy of the GNU General Public License 
15  * along with this program. If not, see <http://www.gnu.org/licenses/>. 
16  */
17
18 #include "DBHelper.h"
19 #include "UserNode.h"
20 #include "ChanNode.h"
21 #include "ChanUser.h"
22 #include "mysqlConn.h"
23 #include "lang.h"
24 #include "tools.h"
25 #include "IRCEvents.h"
26 #include "HandleInfoHandler.h"
27 #include "ClientSocket.h"
28 #include "bots.h"
29 #include "ConfigParser.h"
30
31 void _loadUserSettings(struct UserNode *user) {
32     SYNCHRONIZE(cache_sync);
33     MYSQL_RES *res;
34     MYSQL_ROW row;
35     printf_mysql_query("SELECT `user_lang`, `user_reply_privmsg`, `user_god`, `user_id` FROM `users` WHERE `user_user` = '%s'", escape_string(user->auth));
36     res = mysql_use();
37     if ((row = mysql_fetch_row(res)) != NULL) {
38         user->language = get_language_by_tag(row[0]);
39         if(user->language == NULL) user->language = get_default_language();
40         if(strcmp(row[1], "0"))
41             user->flags |= USERFLAG_REPLY_PRIVMSG;
42         if(strcmp(row[2], "0"))
43             user->flags |= USERFLAG_GOD_MODE;
44         user->user_id = atoi(row[3]);
45         user->flags |= USERFLAG_HAS_USERID;
46     } else
47         user->language = get_default_language();
48     user->flags |= USERFLAG_LOADED_SETTINGS;
49     DESYNCHRONIZE(cache_sync);
50 }
51
52 int isGodMode(struct UserNode *user) {
53     loadUserSettings(user);
54     return (user->flags & USERFLAG_GOD_MODE);
55 }
56
57 int getChannelAccess(struct UserNode *user, struct ChanNode *chan) {
58     if(!(user->flags & USERFLAG_ISAUTHED)) return 0;
59     loadChannelSettings(chan);
60     if(!(chan->flags & CHANFLAG_CHAN_REGISTERED)) return 0;
61     MYSQL_RES *res;
62     MYSQL_ROW row;
63     int caccess = 0;
64     int userid;
65     if(user->flags & USERFLAG_HAS_USERID)
66         userid = user->user_id;
67     else {
68         printf_mysql_query("SELECT `user_id` FROM `users` WHERE `user_user` = '%s'", escape_string(user->auth));
69         res = mysql_use();
70         if ((row = mysql_fetch_row(res)) != NULL) {
71             userid = atoi(row[0]);
72             user->user_id = userid;
73             user->flags |= USERFLAG_HAS_USERID;
74         } else
75             return 0;
76     }
77     printf_mysql_query("SELECT `chanuser_access`, `chanuser_flags` FROM `chanusers` WHERE `chanuser_uid` = '%d' AND `chanuser_cid` = '%d'", userid, chan->channel_id);
78     res = mysql_use();
79     if ((row = mysql_fetch_row(res)) != NULL) {
80         int cflags = atoi(row[1]);
81         if(!(cflags & DB_CHANUSER_SUSPENDED) && atoi(row[0]) > caccess)
82             caccess = atoi(row[0]);
83     }
84     return caccess;
85 }
86
87 char *getChanDefault(char *channel_setting) {
88     MYSQL_RES *res;
89     MYSQL_ROW row;
90     printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_name` = 'defaults'", channel_setting);
91     res = mysql_use();
92     if ((row = mysql_fetch_row(res)) == NULL) return "";
93     return row[0];
94 }
95
96 int checkChannelAccess(struct UserNode *user, struct ChanNode *chan, char *channel_setting, int allow_501) {
97     loadChannelSettings(chan);
98     if(!(chan->flags & CHANFLAG_CHAN_REGISTERED)) return 0;
99     if((user->flags & USERFLAG_ISIRCOP)) return 1;
100     MYSQL_RES *res;
101     MYSQL_ROW row;
102     printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_id` = '%d'", channel_setting, chan->channel_id);
103     res = mysql_use();
104     if ((row = mysql_fetch_row(res)) == NULL) return 0;
105     int require_access = atoi((row[0] ? row[0] : getChanDefault(channel_setting)));
106     if(require_access == 0) return 1;
107     if(!(user->flags & USERFLAG_ISAUTHED)) return 0;
108     int caccess = 0;
109     int userid;
110     if(user->flags & USERFLAG_HAS_USERID)
111         userid = user->user_id;
112     else {
113         printf_mysql_query("SELECT `user_id` FROM `users` WHERE `user_user` = '%s'", escape_string(user->auth));
114         res = mysql_use();
115         if ((row = mysql_fetch_row(res)) != NULL) {
116             userid = atoi(row[0]);
117             user->user_id = userid;
118             user->flags |= USERFLAG_HAS_USERID;
119         } else
120             userid = -1;
121     }
122     if(userid > -1) {
123         printf_mysql_query("SELECT `chanuser_access`, `chanuser_flags` FROM `chanusers` WHERE `chanuser_uid` = '%d' AND `chanuser_cid` = '%d'", userid, chan->channel_id);
124         res = mysql_use();
125         if ((row = mysql_fetch_row(res)) != NULL) {
126             int cflags = atoi(row[1]);
127             if(!(cflags & DB_CHANUSER_SUSPENDED))
128                 caccess = atoi(row[0]);
129         }
130     }
131     if(caccess >= require_access) return 1;
132     if(caccess == 500 && require_access == 501 && allow_501) return 1;
133     return 0;
134 }
135
136 void _loadChannelSettings(struct ChanNode *chan) {
137     SYNCHRONIZE(cache_sync);
138     MYSQL_RES *res;
139     MYSQL_ROW row;
140     printf_mysql_query("SELECT `channel_id` FROM `channels` WHERE `channel_name` = '%s'", escape_string(chan->name));
141     res = mysql_use();
142     if ((row = mysql_fetch_row(res)) != NULL) {
143         chan->flags |= CHANFLAG_CHAN_REGISTERED;
144         chan->channel_id = atoi(row[0]);
145     }
146     chan->flags |= CHANFLAG_REQUESTED_CHANINFO;
147     DESYNCHRONIZE(cache_sync);
148 }
149
150 //TODO: fix performance: we should cache the user access
151 int isUserProtected(struct ChanNode *chan, struct UserNode *victim, struct UserNode *issuer) {
152     /* Don't protect if someone is attacking himself, or if the aggressor is an IRC Operator. */
153     if(victim == issuer || (issuer->flags & USERFLAG_ISIRCOP)) return 0;
154     
155     /* Don't protect if no one is to be protected. */
156     MYSQL_RES *res;
157     MYSQL_ROW row;
158     char protection;
159     loadChannelSettings(chan);
160     if(!(chan->flags & CHANFLAG_CHAN_REGISTERED)) return 0;
161     printf_mysql_query("SELECT `channel_protect` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
162     res = mysql_use();
163     if(!(row = mysql_fetch_row(res))) return 0;
164     if(row[0]) {
165         protection = (char) atoi(row[0]);
166     } else {
167          printf_mysql_query("SELECT `channel_protect` FROM `channels` WHERE `channel_name` = 'defaults'");
168         res = mysql_use();
169         row = mysql_fetch_row(res);
170         protection = (char) atoi(row[0]);
171     }
172     if(protection == 3) return 0;
173     
174     /* Don't protect if the victim isn't added to the channel, unless we are to protect non-users also. */
175     int victim_access = getChannelAccess(victim, chan);
176     if (!victim_access && protection != 0) return 0;
177     
178     /* Protect if the aggressor isn't a user because at this point, the aggressor can only be less than or equal to the victim. */
179     int issuer_access = getChannelAccess(issuer, chan);
180     if (!issuer_access) return 1;
181     
182     /* If the aggressor was a user, then the victim can't be helped. */
183     if(!victim_access) return 0;
184     
185     switch(protection) {
186         case 0:
187         case 1:
188             if(victim_access >= issuer_access) return 1;
189             break;
190         case 2:
191             if(victim_access > issuer_access) return 1;
192             break;
193     }
194     return 0;
195 }
196
197 char *getBanAffectingMask(struct ChanNode *chan, char *mask) {
198     loadChannelSettings(chan);
199     if(!(chan->flags & CHANFLAG_CHAN_REGISTERED)) return 0;
200     MYSQL_RES *res;
201     MYSQL_ROW row;
202     printf_mysql_query("SELECT `ban_mask` FROM `bans` WHERE `ban_channel` = '%d'", chan->channel_id);
203     res = mysql_use();
204     while ((row = mysql_fetch_row(res)) != NULL) {
205         if(!match(row[0], mask))
206             return row[0];
207     }
208     return NULL;
209 }
210
211 int renameAccount(char *oldauth, char *newauth) {
212     MYSQL_RES *res, *res2;
213     MYSQL_ROW row, row2;
214     printf_mysql_query("SELECT `user_id` FROM `users` WHERE `user_user` = '%s'", escape_string(oldauth));
215     res = mysql_use();
216     if ((row = mysql_fetch_row(res)) != NULL) {
217         int userid = atoi(row[0]);
218         printf_mysql_query("SELECT `user_id` FROM `users` WHERE `user_user` = '%s'", escape_string(newauth));
219         res = mysql_use();
220         if((row = mysql_fetch_row(res)) != NULL) {
221             //merge
222             int newuid = atoi(row[0]);
223             printf_mysql_query("SELECT `chanuser_id`, `chanuser_access`, `chanuser_flags` FROM `chanusers` WHERE `chanuser_uid` = '%d'", newuid);
224             res = mysql_use();
225             while((row = mysql_fetch_row(res)) != NULL) {
226                 printf_mysql_query("SELECT `chanuser_id`, `chanuser_access`, `chanuser_flags` FROM `chanusers` WHERE `chanuser_uid` = '%d'", userid);
227                 res2 = mysql_use();
228                 if((row2 = mysql_fetch_row(res2)) != NULL) {
229                     if(atoi(row[0]) > atoi(row2[0])) {
230                         printf_mysql_query("UPDATE `chanusers` SET `chanuser_access` = '%s' WHERE `chanuser_id` = '%s'", row[0], row2[0]);
231                     }
232                     printf_mysql_query("DELETE FROM `chanusers` WHERE `chanuser_id` = '%s'", row[0]);
233                 } else
234                     printf_mysql_query("UPDATE `chanusers` SET `chanuser_uid` = '%d' WHERE `chanuser_id` = '%s'", userid, row[0]);
235             }
236             printf_mysql_query("UPDATE `channels` SET `channel_registrator` = '%d' WHERE `channel_registrator` = '%d'", userid, newuid);
237             printf_mysql_query("UPDATE `bans` SET `ban_owner` = '%d' WHERE `ban_owner` = '%d'", userid, newuid);
238             printf_mysql_query("UPDATE `events` SET `auth` = '%s' WHERE `auth` = '%s'", escape_string(newauth), escape_string(oldauth));
239             printf_mysql_query("UPDATE `godlog` SET `godlog_uid` = '%d' WHERE `godlog_uid` = '%d'", userid, newuid);
240             printf_mysql_query("UPDATE `owner_history` SET `owner_history_to_uid` = '%d' WHERE `owner_history_to_uid` = '%d'", userid, newuid);
241             printf_mysql_query("UPDATE `owner_history` SET `owner_history_from_uid` = '%d' WHERE `owner_history_from_uid` = '%d'", userid, newuid);
242             printf_mysql_query("UPDATE `owner_history` SET `owner_history_from_uid` = '%d' WHERE `owner_history_from_uid` = '%d'", userid, newuid);
243             printf_mysql_query("UPDATE `noinvite` SET `uid` = '%d' WHERE `uid` = '%d'", userid, newuid);
244             printf_mysql_query("DELETE FROM `users` WHERE `chanuser_id` = '%d'", newuid);
245         } else {
246             //simply rename the account
247             printf_mysql_query("UPDATE `users` SET `user_user` = '%s' WHERE `user_id` = '%d'", escape_string(newauth), userid);
248         }
249         char *alertchan = get_string_field("General.CheckAuths.alertchan");
250         if(alertchan) {
251             struct ChanNode *alertchan_chan = getChanByName(alertchan);
252             struct ClientSocket *alertclient;
253             if(alertchan_chan && (alertclient = getChannelBot(alertchan_chan, 0)) != NULL) {
254                 putsock(alertclient, "PRIVMSG %s :Renamed User %s to %s", alertchan_chan->name, oldauth, newauth);
255             }
256         }
257         return 1;
258     }
259     return 0;
260 }
261
262 static AUTHLOOKUP_CALLBACK(event_user_registered_auth_lookup);
263
264 struct event_user_registered_cache {
265     struct UserNode *new_user;
266     char *oldauth;
267 };
268
269 static int event_user_registered(struct UserNode *old_user, struct UserNode *new_user) {
270     //check if there is a fakehost on both sides...
271     if(!isFakeHost(old_user->host) || !isFakeHost(new_user->host)) return 0;
272     //extract user names
273     char oldauth[AUTHLEN], newauth[AUTHLEN];
274     char *p;
275     if((p = strstr(old_user->host, "."))) {
276         *p = '\0';
277         strcpy(oldauth, old_user->host);
278         *p = '.';
279     }
280     if((p = strstr(new_user->host, "."))) {
281         *p = '\0';
282         strcpy(newauth, new_user->host);
283         *p = '.';
284     }
285     if(!stricmp(oldauth, newauth))
286         return 0;
287     //check if we know this user; then check the new auth
288     MYSQL_RES *res;
289     MYSQL_ROW row;
290     printf_mysql_query("SELECT `user_id` FROM `users` WHERE `user_user` = '%s'", escape_string(oldauth));
291     res = mysql_use();
292     if ((row = mysql_fetch_row(res)) != NULL) {
293         struct event_user_registered_cache *cache = malloc(sizeof(*cache));
294         if (!cache) {
295             perror("malloc() failed");
296             return 1;
297         }
298         cache->new_user = new_user;
299         cache->oldauth = strdup(oldauth);
300         lookup_authname(newauth, 0, event_user_registered_auth_lookup, cache);
301     }
302     return 1;
303 }
304
305 static AUTHLOOKUP_CALLBACK(event_user_registered_auth_lookup) {
306     struct event_user_registered_cache *cache = data;
307     if(exists) {
308         renameAccount(cache->oldauth, auth);
309         strcpy(cache->new_user->auth, auth);
310         cache->new_user->flags |= USERFLAG_ISAUTHED;
311     }
312     free(cache->oldauth);
313     free(cache);
314 }
315
316 void deleteUser(int userid) {
317     //simply delete the user
318     MYSQL_RES *res, *res2;
319     MYSQL_ROW row, row2;
320     printf_mysql_query("SELECT a.`chanuser_access`, a.`chanuser_cid`, (SELECT COUNT(*) FROM `chanusers` AS b WHERE b.`chanuser_cid` = a.`chanuser_cid` AND b.`chanuser_access` = 500) FROM `chanusers` AS a WHERE a.`chanuser_uid` = '%d'", userid);
321     res = mysql_use();
322     while((row = mysql_fetch_row(res))) {
323         if(!strcmp(row[0], "500") && !strcmp(row[2], "1")) {
324             //unregister channel
325             printf_mysql_query("SELECT `botid`, `channel_name` FROM `bot_channels` LEFT JOIN `channels` ON `chanid` = `channel_id` WHERE `chanid` = '%s' AND `suspended` = '0'", row[1]);
326             res2 = mysql_use();
327             while((row2 = mysql_fetch_row(res2))) {
328                 struct ClientSocket *bot;
329                 int clientid = atoi(row2[0]);
330                 for(bot = getBots(SOCKET_FLAG_READY, NULL); bot; bot = getBots(SOCKET_FLAG_READY, bot)) {
331                     if(bot->clientid == clientid)
332                         putsock(bot, "PART %s :Channel unregistered.", row2[1]);
333                 }
334             }
335             printf_mysql_query("DELETE FROM `bot_channels` WHERE `chanid` = '%s'", row[1]);
336         }
337     }
338     printf_mysql_query("DELETE FROM `chanusers` WHERE `chanuser_uid` = '%d'", userid);
339     printf_mysql_query("UPDATE `bans` SET `ban_owner` = 0 WHERE `ban_owner` = '%d'", userid);
340     printf_mysql_query("UPDATE `donotregister` SET `dnr_user` = 0 WHERE `dnr_user` = '%d'", userid);
341     printf_mysql_query("UPDATE `bans` SET `ban_owner` = 0 WHERE `ban_owner` = '%d'", userid);
342     printf_mysql_query("UPDATE `godlog` SET `godlog_uid` = 0 WHERE `godlog_uid` = '%d'", userid);
343     printf_mysql_query("DELETE FROM `noinvite` WHERE `uid` = '%d'", userid);
344     printf_mysql_query("UPDATE `owner_history` SET `owner_history_to_uid` = 0 WHERE `owner_history_to_uid` = '%d'", userid);
345     printf_mysql_query("UPDATE `owner_history` SET `owner_history_from_uid` = 0 WHERE `owner_history_from_uid` = '%d'", userid);
346     printf_mysql_query("UPDATE `channels` SET `channel_registrator` = 0 WHERE `channel_registrator` = '%d'", userid);
347     printf_mysql_query("DELETE FROM `users` WHERE `user_id` = '%d'", userid);
348     struct UserNode *user;
349     for(user = getAllUsers(NULL); user; user = getAllUsers(user)) {
350         if(user->flags & USERFLAG_HAS_USERID)
351             user->flags &= ~USERFLAG_HAS_USERID;
352     }
353 }
354
355 void init_DBHelper() {
356     bind_registered(event_user_registered, 0);
357 }
358