2 #include "cmd_neonserv.h"
4 typedef char* neonserv_cmd_set_function(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
5 static void neonserv_cmd_set_setting(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, int setting, char *argument);
6 static char* neonserv_cmd_set_trigger(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
7 static char* neonserv_cmd_set_modes(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
8 static char* neonserv_cmd_set_dynlimit(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
9 static char* neonserv_cmd_set_nodelete(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument);
11 #define NS_VALID_FUNCTION 0x01
12 #define NS_VALID_STRING 0x02
13 #define NS_VALID_ACCESS 0x04
14 #define NS_VALID_NO501 0x08
15 #define NS_VALID_OPTIONS 0x10
16 #define NS_VALID_NUMERIC 0x20
17 #define NS_VALID_BOOLEAN 0x40
19 #define NS_HAS_OPT 0x100 /* options (SET_OPTION_{NAME}_{VALUE}) */
20 #define NS_HAS_HELP 0x200 /* help (SET_HELP_{NAME}) - only shown if help is requested */
24 const char *chanfield;
27 } channel_settings[] = {
28 {"TRIGGER", NULL, NS_VALID_FUNCTION, neonserv_cmd_set_trigger},
29 {"DEFAULTTOPIC", "channel_defaulttopic", NS_VALID_STRING, NULL},
30 {"TOPICMASK", "channel_topicmask", NS_VALID_STRING, NULL},
31 {"ADVANCEDTOPIC", "channel_exttopic", NS_VALID_BOOLEAN | NS_HAS_OPT, NULL},
32 {"GREETING", "channel_greeting", NS_VALID_STRING, NULL},
33 {"USERGREETING", "channel_usergreeting", NS_VALID_STRING, NULL},
34 {"USERINFO", "channel_userinfo", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
35 {"WIPEINFO", "channel_wipeinfo", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
36 {"MODES", "channel_modes", NS_VALID_FUNCTION, neonserv_cmd_set_modes},
37 {"INVITEME", "channel_getinvite", NS_VALID_ACCESS, NULL},
38 {"GIVEOPS", "channel_getop", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
39 {"GIVEVOICE", "channel_getvoice", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
40 {"ENFOPS", "channel_canop", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
41 {"ENFVOICE", "channel_canvoice", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
42 {"KICK", "channel_cankick", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
43 {"BAN", "channel_canban", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
44 {"STATICBAN", "channel_staticban", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
45 {"PUBCMD", "channel_pubcmd", NS_VALID_ACCESS, NULL},
46 {"ENFMODES", "channel_enfmodes", NS_VALID_ACCESS, NULL},
47 {"ENFTOPIC", "channel_enftopic", NS_VALID_ACCESS, NULL},
48 {"TOPICSNARF", "channel_topicsnarf", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
49 {"CHANGETOPIC", "channel_changetopic", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
50 {"SETTERS", "channel_setters", NS_VALID_ACCESS | NS_VALID_NO501 | NS_HAS_HELP, NULL},
51 {"ADDUSER", "channel_canadd", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
52 {"DELUSER", "channel_candel", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
53 {"CLVL", "channel_canclvl", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
54 {"RESYNC", "channel_canresync", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
55 {"SUSPEND", "channel_cansuspend", NS_VALID_ACCESS | NS_HAS_HELP, NULL},
56 {"NOTICEUSERS", "channel_notice", NS_VALID_ACCESS, NULL},
57 {"NOTICEREACTION", "channel_noticereaction", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
58 {"CTCPUSERS", "channel_ctcp", NS_VALID_ACCESS, NULL},
59 {"CTCPREACTION", "channel_ctcpreaction", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
60 {"PROTECT", "channel_protect", NS_VALID_OPTIONS | NS_HAS_OPT, "4"},
61 {"TOYS", "channel_toys", NS_VALID_OPTIONS | NS_HAS_OPT, "3"},
62 {"DYNLIMIT", "channel_dynlimit", NS_VALID_NUMERIC | NS_VALID_FUNCTION | NS_HAS_OPT, neonserv_cmd_set_dynlimit},
63 {"NODELETE", "channel_nodelete", NS_VALID_BOOLEAN | NS_VALID_FUNCTION, neonserv_cmd_set_nodelete},
67 #define MAX_QUERY_LEN 1024
68 CMD_BIND(neonserv_cmd_set) {
70 if(argc && !strcmp(argv[0], "defaults")) {
71 //reset channel settings
72 int uaccess = getChannelAccess(user, chan, 0);
75 event->flags |= CMDFLAG_OPLOG;
77 reply(getTextBot(), user, "NS_SET_DEFAULTS_OWNER", chan->name);
83 static char defaultskey[16];
84 for(tmp = user->auth; *tmp; tmp++)
85 seed = (seed * 0xEECE66DL ^ ((*tmp << 24) | (*tmp << 16) | (*tmp << 8) | *tmp));
86 for(tmp = chan->name; *tmp; tmp++)
87 seed = (seed * 0xEECE66DL ^ ((*tmp << 24) | (*tmp << 16) | (*tmp << 8) | *tmp));
88 sprintf(defaultskey, "%08x", seed);
89 if(argc > 1 && !strcmp(argv[1], defaultskey)) {
90 char query[MAX_QUERY_LEN];
93 while(channel_settings[i].setting) {
94 if(channel_settings[i].chanfield)
95 querypos += sprintf(query + querypos, "`%s` = NULL, ", channel_settings[i].chanfield);
99 query[querypos-2] = '\0';
101 printf_mysql_query("UPDATE `channels` SET %s WHERE `channel_id` = '%d'", query, chan->channel_id);
102 reply(getTextBot(), user, "NS_SET_DEFAULTS_DONE", chan->name);
105 reply(getTextBot(), user, "NS_SET_DEFAULTS_CODE", chan->name, defaultskey);
107 } else if(argc && strcmp(argv[0], "help")) {
108 //find the correct command
111 char *args = (argc > 1 ? merge_argv(argv, 1, argc) : NULL);
112 while(channel_settings[i].setting) {
113 if(!stricmp(channel_settings[i].setting, argv[0])) {
115 if(channel_settings[i].valid & NS_VALID_FUNCTION) {
116 neonserv_cmd_set_function *func = channel_settings[i].parameter;
117 func(client, user, chan, event, channel_settings[i].setting, args);
119 neonserv_cmd_set_setting(client, user, chan, event, i, args);
128 reply(getTextBot(), user, "NS_SET_UNKNOWN_SETTING", argv[0]);
131 char query[MAX_QUERY_LEN], *value, *org_value, *tmp, nameBuf[64];
133 MYSQL_RES *res, *defaults_res;
134 MYSQL_ROW row, defaults;
138 while(channel_settings[i].setting) {
139 if(channel_settings[i].chanfield)
140 querypos += sprintf(query + querypos, ", `%s`", channel_settings[i].chanfield);
143 table = table_init(2, i, 0);
144 table_set_bold(table, 0, 1);
145 printf_mysql_query("SELECT `channel_id` %s FROM `channels` WHERE `channel_name` = 'defaults'", query);
146 defaults_res = mysql_use();
147 defaults = mysql_fetch_row(defaults_res);
148 printf_mysql_query("SELECT `channel_name` %s FROM `channels` WHERE `channel_id` = '%d'", query, chan->channel_id);
150 row = mysql_fetch_row(res);
153 reply(getTextBot(), user, "NS_SET_HEADER", chan->name);
154 while(channel_settings[i].setting) {
155 if(channel_settings[i].chanfield) {
157 org_value = (row[j] ? row[j] : defaults[j]);
158 } else if(channel_settings[i].valid & NS_VALID_FUNCTION) {
159 neonserv_cmd_set_function *func = channel_settings[i].parameter;
160 org_value = func(client, user, chan, event, NULL, NULL);
164 if(channel_settings[i].valid & NS_VALID_BOOLEAN) {
165 if(!strcmp(value, "0"))
166 value = get_language_string(user, "NS_SET_OFF");
168 value = get_language_string(user, "NS_SET_ON");
170 strcpy(query, value);
171 querypos = strlen(query);
172 if(channel_settings[i].valid & NS_HAS_OPT) {
173 sprintf(nameBuf, "NS_SET_OPTION_%s_%s", channel_settings[i].setting, org_value);
174 tmp = get_language_string(user, nameBuf);
176 querypos += sprintf(query+querypos, " - %s", tmp);
179 if(argc && channel_settings[i].valid & NS_HAS_HELP) {
180 sprintf(nameBuf, "NS_SET_HELP_%s", channel_settings[i].setting);
181 tmp = get_language_string(user, nameBuf);
183 querypos += sprintf(query+querypos, " - %s", tmp);
186 content[0] = (char*)channel_settings[i].setting;
188 table_add(table, content);
191 char **table_lines = table_end(table);
192 for(i = 0; i < table->entrys; i++) {
193 reply(getTextBot(), user, table_lines[i]);
199 static void neonserv_cmd_set_setting(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, int setting, char *args) {
205 printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_id` = '%d'", channel_settings[setting].chanfield, chan->channel_id);
207 row = mysql_fetch_row(res);
209 printf_mysql_query("SELECT `%s` FROM `channels` WHERE `channel_name` = 'defaults'", channel_settings[setting].chanfield);
211 row = mysql_fetch_row(res);
215 //change the channel setting
216 //check the new argument
217 int valid = channel_settings[setting].valid;
218 if(valid & NS_VALID_STRING) {
219 if(!strcmp(args, "*")) {
223 if(valid & NS_VALID_ACCESS) {
224 int caccess = atoi(args);
225 int max = ((valid & NS_VALID_NO501) ? 500 : 501);
226 if(caccess < 0 || caccess > max) {
227 reply(getTextBot(), user, "NS_INVALID_ACCESS", caccess);
230 int uaccess = getChannelAccess(user, chan, 0);
231 if(uaccess == 500) uaccess++;
232 if(atoi(value) > uaccess) {
233 if(isGodMode(user)) {
234 event->flags |= CMDFLAG_OPLOG;
236 reply(getTextBot(), user, "NS_SET_CANNOT_SET");
240 if(caccess > uaccess) {
241 if(isGodMode(user)) {
242 event->flags |= CMDFLAG_OPLOG;
244 reply(getTextBot(), user, "NS_SET_BADLEVEL");
248 sprintf(nameBuf, "%d", caccess);
251 if(valid & NS_VALID_OPTIONS) {
252 int options = atoi((char *) channel_settings[setting].parameter);
253 int coption = atoi(args);
254 if(coption < 0 || coption >= options) {
255 reply(getTextBot(), user, "NS_SET_INVALID_OPTION", coption);
258 if(valid & NS_HAS_OPT) {
259 for(i = 0; i < options; i++) {
260 sprintf(nameBuf, "NS_SET_OPTION_%s_%d", channel_settings[setting].setting, i);
261 reply(getTextBot(), user, "\002%d\002 - %s", i, get_language_string(user, nameBuf));
264 for(i = 0; i < options; i++) {
265 nameBufPos += sprintf(nameBuf + nameBufPos, "\002%d\002, ", i);
268 nameBuf[nameBufPos-2] = '\0';
269 reply(getTextBot(), user, nameBuf);
275 if(valid & NS_VALID_NUMERIC) {
276 sprintf(nameBuf, "%d", atoi(args));
279 if(valid & NS_VALID_BOOLEAN) {
280 if(!strcmp(args, "0") || !stricmp(args, "off") || !stricmp(args, get_language_string(user, "NS_SET_OFF"))) {
282 } else if(!strcmp(args, "1") || !stricmp(args, "on") || !stricmp(args, get_language_string(user, "NS_SET_ON"))) {
285 reply(getTextBot(), user, "NS_SET_INVALID_BOOLEAN", args);
291 printf_mysql_query("UPDATE `channels` SET `%s` = '%s' WHERE `channel_id` = '%d'", channel_settings[setting].chanfield, escape_string(value), chan->channel_id);
294 reply(getTextBot(), user, "\002%s\002 %s", channel_settings[setting].setting, value);
295 if(channel_settings[setting].valid & NS_HAS_HELP) {
296 sprintf(nameBuf, "NS_SET_HELP_%s", channel_settings[setting].setting);
297 reply(getTextBot(), user, " %s", get_language_string(user, nameBuf));
301 static char* neonserv_cmd_set_trigger(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
303 //get current trigger
306 printf_mysql_query("SELECT `trigger` FROM `bot_channels` WHERE `chanid` = '%d' AND `botid` = '%d'", chan->channel_id, client->clientid);
308 row = mysql_fetch_row(res);
311 int uaccess = getChannelAccess(user, chan, 0);
313 if(isGodMode(user)) {
314 event->flags |= CMDFLAG_OPLOG;
316 reply(getTextBot(), user, "NS_SET_TRIGGER_OWNER", chan->name);
320 if(strlen(argument) > 15)
322 printf_mysql_query("UPDATE `bot_channels` SET `trigger` = '%s' WHERE `chanid` = '%d' AND `botid` = '%d'", escape_string(argument), chan->channel_id, client->clientid);
324 changeChannelTrigger(client->botid, chan, trigger);
328 reply(getTextBot(), user, "\002%s\002 %s", setting, trigger);
333 static char* neonserv_cmd_set_modes(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
338 printf_mysql_query("SELECT `channel_modes` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
340 row = mysql_fetch_row(res);
342 printf_mysql_query("SELECT `channel_modes` FROM `channels` WHERE `channel_name` = 'defaults'");
344 row = mysql_fetch_row(res);
348 //change the channel setting
349 //TODO: parse, check and set modelock
352 reply(getTextBot(), user, "\002%s\002 %s", setting, value);
357 static char* neonserv_cmd_set_dynlimit(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
363 printf_mysql_query("SELECT `channel_dynlimit` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
365 row = mysql_fetch_row(res);
367 printf_mysql_query("SELECT `channel_dynlimit` FROM `channels` WHERE `channel_name` = 'defaults'");
369 row = mysql_fetch_row(res);
373 //change the channel setting
374 sprintf(tmp, "%d", atoi(argument));
376 printf_mysql_query("UPDATE `channels` SET `channel_dynlimit` = '%s' WHERE `channel_id` = '%d'", escape_string(argument), chan->channel_id);
377 if(strcmp(argument, "0"))
378 putsock(client, "MODE %s +l %d", chan->name, (chan->usercount + atoi(argument)));
379 else if(isModeSet(chan->modes, 'l'))
380 putsock(client, "MODE %s -l", chan->name);
385 reply(getTextBot(), user, "\002%s\002 %s", setting, value);
390 static char* neonserv_cmd_set_nodelete(struct ClientSocket *client, struct UserNode *user, struct ChanNode *chan, struct Event *event, const char *setting, char *argument) {
395 printf_mysql_query("SELECT `channel_nodelete` FROM `channels` WHERE `channel_id` = '%d'", chan->channel_id);
397 row = mysql_fetch_row(res);
399 printf_mysql_query("SELECT `channel_nodelete` FROM `channels` WHERE `channel_name` = 'defaults'");
401 row = mysql_fetch_row(res);
404 if(argument && isGodMode(user)) {
405 //change the channel setting
406 if(!strcmp(argument, "0") || !strcmp(argument, "off") || !strcmp(argument, get_language_string(user, "NS_SET_OFF"))) {
408 } else if(!strcmp(argument, "0") || !strcmp(argument, "off") || !strcmp(argument, get_language_string(user, "NS_SET_OFF"))) {
411 reply(getTextBot(), user, "NS_SET_INVALID_BOOLEAN", argument);
414 printf_mysql_query("UPDATE `channels` SET `channel_nodelete` = '%s' WHERE `channel_id` = '%d'", escape_string(argument), chan->channel_id);
415 event->flags |= CMDFLAG_OPLOG;
420 reply(getTextBot(), user, "\002%s\002 %s", setting, value);