1 /* cmd_neonserv_set.c - NeonServ v5.3
2 * Copyright (C) 2011-2012 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
34 #define NS_VALID_IF_HALFOP 0x80
36 #define NS_HAS_OPT 0x100 /* options (SET_OPTION_{NAME}_{VALUE}) */
37 #define NS_HAS_HELP 0x200 /* help (SET_HELP_{NAME}) - only shown if help is requested */
41 const char *chanfield;
44 } channel_settings[] = {
45 {"TRIGGER", NULL, NS_VALID_FUNCTION, neonserv_cmd_set_trigger},
46 {"DEFAULTTOPIC", "channel_defaulttopic", NS_VALID_STRING, NULL},
47 {"TOPICMASK", "channel_topicmask", NS_VALID_STRING, NULL},
48 {"ADVANCEDTOPIC", "channel_exttopic", NS_VALID_BOOLEAN | NS_HAS_OPT, NULL},
49 {"GREETING", "channel_greeting", NS_VALID_STRING, NULL},
50 {"USERGREETING", "channel_usergreeting", NS_VALID_STRING, NULL},
51 {"USERINFO", "channel_userinfo", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
52 {"WIPEINFO", "channel_wipeinfo", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
53 {"MODES", "channel_modes", NS_VALID_FUNCTION, neonserv_cmd_set_modes},
54 {"INVITEME", "channel_getinvite", NS_VALID_ACCESS, NULL},
55 {"GIVEOPS", "channel_getop", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
56 {"GIVEHALFOPS", "channel_gethalfop", NS_VALID_ACCESS | NS_HAS_HELP | NS_VALID_IF_HALFOP, NULL},
57 {"GIVEVOICE", "channel_getvoice", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
58 {"ENFOPS", "channel_canop", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
59 {"ENFHALFOPS", "channel_canhalfop", NS_VALID_ACCESS | NS_HAS_HELP | NS_VALID_IF_HALFOP, NULL},
60 {"ENFVOICE", "channel_canvoice", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
61 {"KICK", "channel_cankick", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
62 {"BAN", "channel_canban", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
63 {"STATICBAN", "channel_staticban", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
64 {"PUBCMD", "channel_pubcmd", NS_VALID_ACCESS, NULL},
65 {"ENFMODES", "channel_enfmodes", NS_VALID_ACCESS, NULL},
66 {"ENFTOPIC", "channel_enftopic", NS_VALID_ACCESS, NULL},
67 {"TOPICSNARF", "channel_topicsnarf", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
68 {"CHANGETOPIC", "channel_changetopic", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
69 {"SETTERS", "channel_setters", NS_VALID_ACCESS | NS_VALID_NO501 | NS_HAS_HELP, NULL},
70 {"ADDUSER", "channel_canadd", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
71 {"DELUSER", "channel_candel", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
72 {"CLVL", "channel_canclvl", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
73 {"RESYNC", "channel_canresync", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
74 {"SUSPEND", "channel_cansuspend", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
75 {"NOTICEUSERS", "channel_notice", NS_VALID_ACCESS, NULL},
76 {"NOTICEREACTION", "channel_noticereaction", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
77 {"CTCPUSERS", "channel_ctcp", NS_VALID_ACCESS, NULL},
78 {"CTCPREACTION", "channel_ctcpreaction", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
79 {"PROTECT", "channel_protect", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
80 {"TOYS", "channel_toys", NS_VALID_OPTIONS | NS_HAS_OPT, "3"},
81 {"DYNLIMIT", "channel_dynlimit", NS_VALID_NUMERIC | NS_VALID_FUNCTION | NS_HAS_OPT, neonserv_cmd_set_dynlimit},
82 {"NODELETE", "channel_nodelete", NS_VALID_BOOLEAN | NS_VALID_FUNCTION, neonserv_cmd_set_nodelete},
86 #define MAX_QUERY_LEN 1024
87 CMD_BIND(neonserv_cmd_set) {
89 if(argc && !strcmp(argv[0], "defaults")) {
90 //reset channel settings
91 int uaccess = getChannelAccess(user, chan);
94 event->flags |= CMDFLAG_OPLOG;
96 reply(getTextBot(), user, "NS_SET_DEFAULTS_OWNER", chan->name);
102 static char defaultskey[16];
103 for(tmp = user->auth; *tmp; tmp++)
104 seed = (seed * 0xEECE66DL ^ ((*tmp << 24) | (*tmp << 16) | (*tmp << 8) | *tmp));
105 for(tmp = chan->name; *tmp; tmp++)
106 seed = (seed * 0xEECE66DL ^ ((*tmp << 24) | (*tmp << 16) | (*tmp << 8) | *tmp));
107 sprintf(defaultskey, "%08x", seed);
108 if(argc > 1 && !strcmp(argv[1], defaultskey)) {
109 char query[MAX_QUERY_LEN];
112 while(channel_settings[i].setting) {
113 if(channel_settings[i].chanfield)
114 querypos += sprintf(query + querypos, "`%s` = NULL, ", channel_settings[i].chanfield);
118 query[querypos-2] = '\0';
120 printf_mysql_query("UPDATE `channels` SET %s WHERE `channel_id` = '%d'", query, chan->channel_id);
121 reply(getTextBot(), user, "NS_SET_DEFAULTS_DONE", chan->name);
124 reply(getTextBot(), user, "NS_SET_DEFAULTS_CODE", chan->name, defaultskey);
126 } else if(argc && strcmp(argv[0], "help")) {
127 //find the correct command
130 char *args = (argc > 1 ? merge_argv(argv, 1, argc) : NULL);
131 int with_halfops = get_int_field("General.have_halfop");
132 while(channel_settings[i].setting) {
133 if(!stricmp(channel_settings[i].setting, argv[0]) && (!(channel_settings[i].valid & NS_VALID_IF_HALFOP) || with_halfops)) {
135 if(channel_settings[i].valid & NS_VALID_FUNCTION) {
136 neonserv_cmd_set_function *func = channel_settings[i].parameter;
137 func(client, user, chan, event, channel_settings[i].setting, args);
139 neonserv_cmd_set_setting(client, user, chan, event, i, args);
148 reply(getTextBot(), user, "NS_SET_UNKNOWN_SETTING", argv[0]);
151 char query[MAX_QUERY_LEN], *value, *org_value, *tmp, nameBuf[64];
153 MYSQL_RES *res, *defaults_res;
154 MYSQL_ROW row, defaults;
157 int with_halfops = get_int_field("General.have_halfop");
160 while(channel_settings[i].setting) {
161 if((channel_settings[i].valid & NS_VALID_IF_HALFOP) && !with_halfops) {
165 if(channel_settings[i].chanfield)
166 querypos += sprintf(query + querypos, ", `%s`", channel_settings[i].chanfield);
169 table = table_init(2, i-j, 0);
170 table_set_bold(table, 0, 1);
171 printf_mysql_query("SELECT `channel_id` %s FROM `channels` WHERE `channel_name` = 'defaults'", query);
172 defaults_res = mysql_use();
173 defaults = mysql_fetch_row(defaults_res);
174 printf_mysql_query("SELECT `channel_name` %s FROM `channels` WHERE `channel_id` = '%d'", query, chan->channel_id);
176 row = mysql_fetch_row(res);
179 reply(getTextBot(), user, "NS_SET_HEADER", chan->name);
180 while(channel_settings[i].setting) {
181 if((channel_settings[i].valid & NS_VALID_IF_HALFOP) && !with_halfops) {
185 if(channel_settings[i].chanfield) {
187 org_value = (row[j] ? row[j] : defaults[j]);
188 } else if(channel_settings[i].valid & NS_VALID_FUNCTION) {
189 neonserv_cmd_set_function *func = channel_settings[i].parameter;
190 org_value = func(client, user, chan, event, NULL, NULL);
194 if(channel_settings[i].valid & NS_VALID_BOOLEAN) {
195 if(!strcmp(value, "0"))
196 value = get_language_string(user, "NS_SET_OFF");
198 value = get_language_string(user, "NS_SET_ON");
200 strcpy(query, value);
201 querypos = strlen(query);
202 if(channel_settings[i].valid & NS_HAS_OPT) {
203 sprintf(nameBuf, "NS_SET_OPTION_%s_%s", channel_settings[i].setting, org_value);
204 tmp = get_language_string(user, nameBuf);
206 querypos += sprintf(query+querypos, " - %s", tmp);
209 if(argc && channel_settings[i].valid & NS_HAS_HELP) {
210 sprintf(nameBuf, "NS_SET_HELP_%s", channel_settings[i].setting);
211 tmp = get_language_string(user, nameBuf);
213 querypos += sprintf(query+querypos, " - %s", tmp);
216 content[0] = (char*)channel_settings[i].setting;
218 table_add(table, content);
221 char **table_lines = table_end(table);
222 for(i = 0; i < table->entrys; i++) {
223 reply(getTextBot(), user, table_lines[i]);
229 static void neonserv_cmd_set_setting(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, int setting, char *args) {
235 printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_id` = '%d'", channel_settings[setting].chanfield, chan->channel_id);
237 row = mysql_fetch_row(res);
239 printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_name` = 'defaults'", channel_settings[setting].chanfield);
241 row = mysql_fetch_row(res);
245 //change the channel setting
246 //check the new argument
247 int valid = channel_settings[setting].valid;
248 if(valid & NS_VALID_STRING) {
249 if(!strcmp(args, "*")) {
253 if(valid & NS_VALID_ACCESS) {
254 int caccess = atoi(args);
255 int max = ((valid & NS_VALID_NO501) ? 500 : 501);
256 if(caccess < 0 || caccess > max) {
257 reply(getTextBot(), user, "NS_INVALID_ACCESS", caccess);
260 int uaccess = getChannelAccess(user, chan);
261 if(uaccess == 500) uaccess++;
262 if(atoi(value) > uaccess) {
263 if(isGodMode(user)) {
264 event->flags |= CMDFLAG_OPLOG;
266 reply(getTextBot(), user, "NS_SET_CANNOT_SET");
270 if(caccess > uaccess) {
271 if(isGodMode(user)) {
272 event->flags |= CMDFLAG_OPLOG;
274 reply(getTextBot(), user, "NS_SET_BADLEVEL");
278 sprintf(nameBuf, "%d", caccess);
281 if(valid & NS_VALID_OPTIONS) {
282 int options = atoi((char *) channel_settings[setting].parameter);
283 int coption = atoi(args);
284 if(coption < 0 || coption >= options) {
285 reply(getTextBot(), user, "NS_SET_INVALID_OPTION", coption);
288 if(valid & NS_HAS_OPT) {
289 for(i = 0; i < options; i++) {
290 sprintf(nameBuf, "NS_SET_OPTION_%s_%d", channel_settings[setting].setting, i);
291 reply(getTextBot(), user, "\002%d\002 - %s", i, get_language_string(user, nameBuf));
294 for(i = 0; i < options; i++) {
295 nameBufPos += sprintf(nameBuf + nameBufPos, "\002%d\002, ", i);
298 nameBuf[nameBufPos-2] = '\0';
299 reply(getTextBot(), user, nameBuf);
305 if(valid & NS_VALID_NUMERIC) {
306 sprintf(nameBuf, "%d", atoi(args));
309 if(valid & NS_VALID_BOOLEAN) {
310 if(!strcmp(args, "0") || !stricmp(args, "off") || !stricmp(args, get_language_string(user, "NS_SET_OFF"))) {
312 } else if(!strcmp(args, "1") || !stricmp(args, "on") || !stricmp(args, get_language_string(user, "NS_SET_ON"))) {
315 reply(getTextBot(), user, "NS_SET_INVALID_BOOLEAN", args);
321 printf_mysql_query("UPDATE `channels` SET `%s` = '%s' WHERE `channel_id` = '%d'", channel_settings[setting].chanfield, escape_string(value), chan->channel_id);
324 if(channel_settings[setting].valid & NS_HAS_OPT) {
325 sprintf(nameBuf, "NS_SET_OPTION_%s_%s", channel_settings[setting].setting, value);
326 char *tmp = get_language_string(user, nameBuf);
328 reply(getTextBot(), user, "\002%s\002 %s - %s", channel_settings[setting].setting, value, tmp);
330 reply(getTextBot(), user, "\002%s\002 %s", channel_settings[setting].setting, value);
332 reply(getTextBot(), user, "\002%s\002 %s", channel_settings[setting].setting, value);
333 if(channel_settings[setting].valid & NS_HAS_HELP) {
334 sprintf(nameBuf, "NS_SET_HELP_%s", channel_settings[setting].setting);
335 reply(getTextBot(), user, " %s", get_language_string(user, nameBuf));
339 static char* neonserv_cmd_set_trigger(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
341 //get current trigger
344 printf_mysql_query("SELECT `trigger`, `defaulttrigger` FROM `bot_channels` LEFT JOIN `bots` ON `botid` = `bots`.`id` WHERE `chanid` = '%d' AND `botid` = '%d'", chan->channel_id, client->clientid);
346 row = mysql_fetch_row(res);
347 trigger = (row[0] ? row[0] : row[1]);
349 int uaccess = getChannelAccess(user, chan);
351 if(isGodMode(user)) {
352 event->flags |= CMDFLAG_OPLOG;
354 reply(getTextBot(), user, "NS_SET_TRIGGER_OWNER", chan->name);
358 if(strlen(argument) > 15)
360 printf_mysql_query("UPDATE `bot_channels` SET `trigger` = '%s' WHERE `chanid` = '%d' AND `botid` = '%d'", escape_string(argument), chan->channel_id, client->clientid);
362 changeChannelTrigger(client->botid, chan, trigger);
366 reply(getTextBot(), user, "\002%s\002 %s", setting, trigger);
371 static char* neonserv_cmd_set_modes(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
373 char valueBuf[MAXLEN];
377 printf_mysql_query("SELECT `channel_modes` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
379 row = mysql_fetch_row(res);
381 printf_mysql_query("SELECT `channel_modes` FROM `channels` WHERE `channel_name` = 'defaults'");
383 row = mysql_fetch_row(res);
387 //change the channel setting
388 struct ModeNode *modenode = createModeNode(NULL);
389 parseModeString(modenode, argument);
390 getFullModeString(modenode, valueBuf);
392 printf_mysql_query("UPDATE `channels` SET `channel_modes` = '%s' WHERE `channel_id` = '%d'", escape_string(value), chan->channel_id);
394 freeModeNode(modenode);
397 reply(getTextBot(), user, "\002%s\002 %s", setting, value);
402 static char* neonserv_cmd_set_dynlimit(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
408 printf_mysql_query("SELECT `channel_dynlimit` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
410 row = mysql_fetch_row(res);
412 printf_mysql_query("SELECT `channel_dynlimit` FROM `channels` WHERE `channel_name` = 'defaults'");
414 row = mysql_fetch_row(res);
418 //change the channel setting
419 sprintf(tmp, "%d", atoi(argument));
421 printf_mysql_query("UPDATE `channels` SET `channel_dynlimit` = '%s' WHERE `channel_id` = '%d'", escape_string(argument), chan->channel_id);
422 if(strcmp(argument, "0"))
423 putsock(client, "MODE %s +l %d", chan->name, (chan->usercount + atoi(argument)));
424 else if(isModeSet(chan->modes, 'l'))
425 putsock(client, "MODE %s -l", chan->name);
430 reply(getTextBot(), user, "\002%s\002 %s", setting, value);
435 static char* neonserv_cmd_set_nodelete(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
440 printf_mysql_query("SELECT `channel_nodelete` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
442 row = mysql_fetch_row(res);
444 printf_mysql_query("SELECT `channel_nodelete` FROM `channels` WHERE `channel_name` = 'defaults'");
446 row = mysql_fetch_row(res);
449 if(argument && isGodMode(user)) {
450 //change the channel setting
451 if(!strcmp(argument, "0") || !strcmp(argument, "off") || !strcmp(argument, get_language_string(user, "NS_SET_OFF"))) {
453 } else if(!strcmp(argument, "0") || !strcmp(argument, "off") || !strcmp(argument, get_language_string(user, "NS_SET_OFF"))) {
456 reply(getTextBot(), user, "NS_SET_INVALID_BOOLEAN", argument);
459 printf_mysql_query("UPDATE `channels` SET `channel_nodelete` = '%s' WHERE `channel_id` = '%d'", escape_string(argument), chan->channel_id);
460 event->flags |= CMDFLAG_OPLOG;
465 reply(getTextBot(), user, "\002%s\002 %s", setting, value);