1 /* cmd_neonserv_set.c - NeonServ v5.0
2 * Copyright (C) 2011 Philipp Kreil (pk910)
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.
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.
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/>.
18 #include "cmd_neonserv.h"
20 typedef char* neonserv_cmd_set_function(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
21 static void neonserv_cmd_set_setting(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, int setting, char *argument);
22 static char* neonserv_cmd_set_trigger(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
23 static char* neonserv_cmd_set_modes(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
24 static char* neonserv_cmd_set_dynlimit(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
25 static char* neonserv_cmd_set_nodelete(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
27 #define NS_VALID_FUNCTION 0x01
28 #define NS_VALID_STRING 0x02
29 #define NS_VALID_ACCESS 0x04
30 #define NS_VALID_NO501 0x08
31 #define NS_VALID_OPTIONS 0x10
32 #define NS_VALID_NUMERIC 0x20
33 #define NS_VALID_BOOLEAN 0x40
35 #define NS_HAS_OPT 0x100 /* options (SET_OPTION_{NAME}_{VALUE}) */
36 #define NS_HAS_HELP 0x200 /* help (SET_HELP_{NAME}) - only shown if help is requested */
40 const char *chanfield;
43 } channel_settings[] = {
44 {"TRIGGER", NULL, NS_VALID_FUNCTION, neonserv_cmd_set_trigger},
45 {"DEFAULTTOPIC", "channel_defaulttopic", NS_VALID_STRING, NULL},
46 {"TOPICMASK", "channel_topicmask", NS_VALID_STRING, NULL},
47 {"ADVANCEDTOPIC", "channel_exttopic", NS_VALID_BOOLEAN | NS_HAS_OPT, NULL},
48 {"GREETING", "channel_greeting", NS_VALID_STRING, NULL},
49 {"USERGREETING", "channel_usergreeting", NS_VALID_STRING, NULL},
50 {"USERINFO", "channel_userinfo", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
51 {"WIPEINFO", "channel_wipeinfo", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
52 {"MODES", "channel_modes", NS_VALID_FUNCTION, neonserv_cmd_set_modes},
53 {"INVITEME", "channel_getinvite", NS_VALID_ACCESS, NULL},
54 {"GIVEOPS", "channel_getop", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
55 {"GIVEVOICE", "channel_getvoice", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
56 {"ENFOPS", "channel_canop", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
57 {"ENFVOICE", "channel_canvoice", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
58 {"KICK", "channel_cankick", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
59 {"BAN", "channel_canban", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
60 {"STATICBAN", "channel_staticban", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
61 {"PUBCMD", "channel_pubcmd", NS_VALID_ACCESS, NULL},
62 {"ENFMODES", "channel_enfmodes", NS_VALID_ACCESS, NULL},
63 {"ENFTOPIC", "channel_enftopic", NS_VALID_ACCESS, NULL},
64 {"TOPICSNARF", "channel_topicsnarf", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
65 {"CHANGETOPIC", "channel_changetopic", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
66 {"SETTERS", "channel_setters", NS_VALID_ACCESS | NS_VALID_NO501 | NS_HAS_HELP, NULL},
67 {"ADDUSER", "channel_canadd", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
68 {"DELUSER", "channel_candel", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
69 {"CLVL", "channel_canclvl", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
70 {"RESYNC", "channel_canresync", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
71 {"SUSPEND", "channel_cansuspend", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
72 {"NOTICEUSERS", "channel_notice", NS_VALID_ACCESS, NULL},
73 {"NOTICEREACTION", "channel_noticereaction", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
74 {"CTCPUSERS", "channel_ctcp", NS_VALID_ACCESS, NULL},
75 {"CTCPREACTION", "channel_ctcpreaction", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
76 {"PROTECT", "channel_protect", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
77 {"TOYS", "channel_toys", NS_VALID_OPTIONS | NS_HAS_OPT, "3"},
78 {"DYNLIMIT", "channel_dynlimit", NS_VALID_NUMERIC | NS_VALID_FUNCTION | NS_HAS_OPT, neonserv_cmd_set_dynlimit},
79 {"NODELETE", "channel_nodelete", NS_VALID_BOOLEAN | NS_VALID_FUNCTION, neonserv_cmd_set_nodelete},
83 #define MAX_QUERY_LEN 1024
84 CMD_BIND(neonserv_cmd_set) {
86 if(argc && !strcmp(argv[0], "defaults")) {
87 //reset channel settings
88 int uaccess = getChannelAccess(user, chan, 0);
91 event->flags |= CMDFLAG_OPLOG;
93 reply(getTextBot(), user, "NS_SET_DEFAULTS_OWNER", chan->name);
99 static char defaultskey[16];
100 for(tmp = user->auth; *tmp; tmp++)
101 seed = (seed * 0xEECE66DL ^ ((*tmp << 24) | (*tmp << 16) | (*tmp << 8) | *tmp));
102 for(tmp = chan->name; *tmp; tmp++)
103 seed = (seed * 0xEECE66DL ^ ((*tmp << 24) | (*tmp << 16) | (*tmp << 8) | *tmp));
104 sprintf(defaultskey, "%08x", seed);
105 if(argc > 1 && !strcmp(argv[1], defaultskey)) {
106 char query[MAX_QUERY_LEN];
109 while(channel_settings[i].setting) {
110 if(channel_settings[i].chanfield)
111 querypos += sprintf(query + querypos, "`%s` = NULL, ", channel_settings[i].chanfield);
115 query[querypos-2] = '\0';
117 printf_mysql_query("UPDATE `channels` SET %s WHERE `channel_id` = '%d'", query, chan->channel_id);
118 reply(getTextBot(), user, "NS_SET_DEFAULTS_DONE", chan->name);
121 reply(getTextBot(), user, "NS_SET_DEFAULTS_CODE", chan->name, defaultskey);
123 } else if(argc && strcmp(argv[0], "help")) {
124 //find the correct command
127 char *args = (argc > 1 ? merge_argv(argv, 1, argc) : NULL);
128 while(channel_settings[i].setting) {
129 if(!stricmp(channel_settings[i].setting, argv[0])) {
131 if(channel_settings[i].valid & NS_VALID_FUNCTION) {
132 neonserv_cmd_set_function *func = channel_settings[i].parameter;
133 func(client, user, chan, event, channel_settings[i].setting, args);
135 neonserv_cmd_set_setting(client, user, chan, event, i, args);
144 reply(getTextBot(), user, "NS_SET_UNKNOWN_SETTING", argv[0]);
147 char query[MAX_QUERY_LEN], *value, *org_value, *tmp, nameBuf[64];
149 MYSQL_RES *res, *defaults_res;
150 MYSQL_ROW row, defaults;
154 while(channel_settings[i].setting) {
155 if(channel_settings[i].chanfield)
156 querypos += sprintf(query + querypos, ", `%s`", channel_settings[i].chanfield);
159 table = table_init(2, i, 0);
160 table_set_bold(table, 0, 1);
161 printf_mysql_query("SELECT `channel_id` %s FROM `channels` WHERE `channel_name` = 'defaults'", query);
162 defaults_res = mysql_use();
163 defaults = mysql_fetch_row(defaults_res);
164 printf_mysql_query("SELECT `channel_name` %s FROM `channels` WHERE `channel_id` = '%d'", query, chan->channel_id);
166 row = mysql_fetch_row(res);
169 reply(getTextBot(), user, "NS_SET_HEADER", chan->name);
170 while(channel_settings[i].setting) {
171 if(channel_settings[i].chanfield) {
173 org_value = (row[j] ? row[j] : defaults[j]);
174 } else if(channel_settings[i].valid & NS_VALID_FUNCTION) {
175 neonserv_cmd_set_function *func = channel_settings[i].parameter;
176 org_value = func(client, user, chan, event, NULL, NULL);
180 if(channel_settings[i].valid & NS_VALID_BOOLEAN) {
181 if(!strcmp(value, "0"))
182 value = get_language_string(user, "NS_SET_OFF");
184 value = get_language_string(user, "NS_SET_ON");
186 strcpy(query, value);
187 querypos = strlen(query);
188 if(channel_settings[i].valid & NS_HAS_OPT) {
189 sprintf(nameBuf, "NS_SET_OPTION_%s_%s", channel_settings[i].setting, org_value);
190 tmp = get_language_string(user, nameBuf);
192 querypos += sprintf(query+querypos, " - %s", tmp);
195 if(argc && channel_settings[i].valid & NS_HAS_HELP) {
196 sprintf(nameBuf, "NS_SET_HELP_%s", channel_settings[i].setting);
197 tmp = get_language_string(user, nameBuf);
199 querypos += sprintf(query+querypos, " - %s", tmp);
202 content[0] = (char*)channel_settings[i].setting;
204 table_add(table, content);
207 char **table_lines = table_end(table);
208 for(i = 0; i < table->entrys; i++) {
209 reply(getTextBot(), user, table_lines[i]);
215 static void neonserv_cmd_set_setting(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, int setting, char *args) {
221 printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_id` = '%d'", channel_settings[setting].chanfield, chan->channel_id);
223 row = mysql_fetch_row(res);
225 printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_name` = 'defaults'", channel_settings[setting].chanfield);
227 row = mysql_fetch_row(res);
231 //change the channel setting
232 //check the new argument
233 int valid = channel_settings[setting].valid;
234 if(valid & NS_VALID_STRING) {
235 if(!strcmp(args, "*")) {
239 if(valid & NS_VALID_ACCESS) {
240 int caccess = atoi(args);
241 int max = ((valid & NS_VALID_NO501) ? 500 : 501);
242 if(caccess < 0 || caccess > max) {
243 reply(getTextBot(), user, "NS_INVALID_ACCESS", caccess);
246 int uaccess = getChannelAccess(user, chan, 0);
247 if(uaccess == 500) uaccess++;
248 if(atoi(value) > uaccess) {
249 if(isGodMode(user)) {
250 event->flags |= CMDFLAG_OPLOG;
252 reply(getTextBot(), user, "NS_SET_CANNOT_SET");
256 if(caccess > uaccess) {
257 if(isGodMode(user)) {
258 event->flags |= CMDFLAG_OPLOG;
260 reply(getTextBot(), user, "NS_SET_BADLEVEL");
264 sprintf(nameBuf, "%d", caccess);
267 if(valid & NS_VALID_OPTIONS) {
268 int options = atoi((char *) channel_settings[setting].parameter);
269 int coption = atoi(args);
270 if(coption < 0 || coption >= options) {
271 reply(getTextBot(), user, "NS_SET_INVALID_OPTION", coption);
274 if(valid & NS_HAS_OPT) {
275 for(i = 0; i < options; i++) {
276 sprintf(nameBuf, "NS_SET_OPTION_%s_%d", channel_settings[setting].setting, i);
277 reply(getTextBot(), user, "\002%d\002 - %s", i, get_language_string(user, nameBuf));
280 for(i = 0; i < options; i++) {
281 nameBufPos += sprintf(nameBuf + nameBufPos, "\002%d\002, ", i);
284 nameBuf[nameBufPos-2] = '\0';
285 reply(getTextBot(), user, nameBuf);
291 if(valid & NS_VALID_NUMERIC) {
292 sprintf(nameBuf, "%d", atoi(args));
295 if(valid & NS_VALID_BOOLEAN) {
296 if(!strcmp(args, "0") || !stricmp(args, "off") || !stricmp(args, get_language_string(user, "NS_SET_OFF"))) {
298 } else if(!strcmp(args, "1") || !stricmp(args, "on") || !stricmp(args, get_language_string(user, "NS_SET_ON"))) {
301 reply(getTextBot(), user, "NS_SET_INVALID_BOOLEAN", args);
307 printf_mysql_query("UPDATE `channels` SET `%s` = '%s' WHERE `channel_id` = '%d'", channel_settings[setting].chanfield, escape_string(value), chan->channel_id);
310 reply(getTextBot(), user, "\002%s\002 %s", channel_settings[setting].setting, value);
311 if(channel_settings[setting].valid & NS_HAS_HELP) {
312 sprintf(nameBuf, "NS_SET_HELP_%s", channel_settings[setting].setting);
313 reply(getTextBot(), user, " %s", get_language_string(user, nameBuf));
317 static char* neonserv_cmd_set_trigger(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
319 //get current trigger
322 printf_mysql_query("SELECT `trigger` FROM `bot_channels` WHERE `chanid` = '%d' AND `botid` = '%d'", chan->channel_id, client->clientid);
324 row = mysql_fetch_row(res);
327 int uaccess = getChannelAccess(user, chan, 0);
329 if(isGodMode(user)) {
330 event->flags |= CMDFLAG_OPLOG;
332 reply(getTextBot(), user, "NS_SET_TRIGGER_OWNER", chan->name);
336 if(strlen(argument) > 15)
338 printf_mysql_query("UPDATE `bot_channels` SET `trigger` = '%s' WHERE `chanid` = '%d' AND `botid` = '%d'", escape_string(argument), chan->channel_id, client->clientid);
340 changeChannelTrigger(client->botid, chan, trigger);
344 reply(getTextBot(), user, "\002%s\002 %s", setting, trigger);
349 static char* neonserv_cmd_set_modes(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
351 char valueBuf[MAXLEN];
355 printf_mysql_query("SELECT `channel_modes` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
357 row = mysql_fetch_row(res);
359 printf_mysql_query("SELECT `channel_modes` FROM `channels` WHERE `channel_name` = 'defaults'");
361 row = mysql_fetch_row(res);
365 //change the channel setting
366 struct ModeNode *modenode = createModeNode(NULL);
367 parseModeString(modeNode, argument);
368 getFullModeString(modeNode, valueBuf);
370 printf_mysql_query("UPDATE `channels` SET `channel_modes` = '%s' WHERE `channel_id` = '%d'", escape_string(value), chan->channel_id);
372 freeModeNode(modenode);
375 reply(getTextBot(), user, "\002%s\002 %s", setting, value);
380 static char* neonserv_cmd_set_dynlimit(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
386 printf_mysql_query("SELECT `channel_dynlimit` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
388 row = mysql_fetch_row(res);
390 printf_mysql_query("SELECT `channel_dynlimit` FROM `channels` WHERE `channel_name` = 'defaults'");
392 row = mysql_fetch_row(res);
396 //change the channel setting
397 sprintf(tmp, "%d", atoi(argument));
399 printf_mysql_query("UPDATE `channels` SET `channel_dynlimit` = '%s' WHERE `channel_id` = '%d'", escape_string(argument), chan->channel_id);
400 if(strcmp(argument, "0"))
401 putsock(client, "MODE %s +l %d", chan->name, (chan->usercount + atoi(argument)));
402 else if(isModeSet(chan->modes, 'l'))
403 putsock(client, "MODE %s -l", chan->name);
408 reply(getTextBot(), user, "\002%s\002 %s", setting, value);
413 static char* neonserv_cmd_set_nodelete(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
418 printf_mysql_query("SELECT `channel_nodelete` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
420 row = mysql_fetch_row(res);
422 printf_mysql_query("SELECT `channel_nodelete` FROM `channels` WHERE `channel_name` = 'defaults'");
424 row = mysql_fetch_row(res);
427 if(argument && isGodMode(user)) {
428 //change the channel setting
429 if(!strcmp(argument, "0") || !strcmp(argument, "off") || !strcmp(argument, get_language_string(user, "NS_SET_OFF"))) {
431 } else if(!strcmp(argument, "0") || !strcmp(argument, "off") || !strcmp(argument, get_language_string(user, "NS_SET_OFF"))) {
434 reply(getTextBot(), user, "NS_SET_INVALID_BOOLEAN", argument);
437 printf_mysql_query("UPDATE `channels` SET `channel_nodelete` = '%s' WHERE `channel_id` = '%d'", escape_string(argument), chan->channel_id);
438 event->flags |= CMDFLAG_OPLOG;
443 reply(getTextBot(), user, "\002%s\002 %s", setting, value);