/* * uhub - A tiny ADC p2p connection hub * Copyright (C) 2007-2011, Jan Vidar Krey * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * */ #include "uhub.h" #include "plugin_api/command_api.h" struct plugin_callback_data { struct linked_list* commands; }; static struct plugin_callback_data* get_callback_data(struct plugin_handle* plugin) { return get_internals(plugin)->callback_data; } static int plugin_command_dispatch(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd) { struct plugin_handle* plugin = (struct plugin_handle*) cmd->ptr; struct plugin_callback_data* data = get_callback_data(plugin); struct plugin_command_handle* cmdh; struct plugin_user* puser = (struct plugin_user*) user; // FIXME: Use a proper conversion function instead. struct plugin_command* pcommand = (struct plugin_command*) cmd; // FIXME: Use a proper conversion function instead. LOG_PLUGIN("plugin_command_dispatch: cmd=%s", cmd->prefix); cmdh = (struct plugin_command_handle*) list_get_first(data->commands); while (cmdh) { if (strcmp(cmdh->prefix, cmd->prefix) == 0) return cmdh->handler(plugin, puser, pcommand); cmdh = (struct plugin_command_handle*) list_get_next(data->commands); } return 0; } static struct hub_user* convert_user_type(struct plugin_handle* plugin, struct plugin_user* user) { /* The plugin_user is not guaranteed to point at the same memory as the * corresponding hub_user - for example, get_user_list() makes a copy of * the data in case the user quits before the plugin uses the list. Hence * we need to look it up by SID. */ struct hub_info* hub = plugin_get_hub(plugin); return uman_get_user_by_sid(hub, user->sid); } static int cbfunc_send_message(struct plugin_handle* plugin, struct plugin_user* user, const char* message) { struct hub_user* huser = convert_user_type(plugin, user); if(huser == NULL) return 0; char* buffer = adc_msg_escape(message); struct adc_message* command = adc_msg_construct(ADC_CMD_IMSG, strlen(buffer) + 6); adc_msg_add_argument(command, buffer); route_to_user(plugin_get_hub(plugin), huser, command); adc_msg_free(command); hub_free(buffer); return 1; } static int cbfunc_send_status(struct plugin_handle* plugin, struct plugin_user* user, int code, const char* message) { struct hub_user* huser = convert_user_type(plugin, user); if(huser == NULL) return 0; char code_str[4]; char* buffer = adc_msg_escape(message); struct adc_message* command = adc_msg_construct(ADC_CMD_ISTA, strlen(buffer) + 10); snprintf(code_str, sizeof(code_str), "%03d", code); adc_msg_add_argument(command, code_str); adc_msg_add_argument(command, buffer); route_to_user(plugin_get_hub(plugin), huser, command); adc_msg_free(command); hub_free(buffer); return 1; } static int cbfunc_user_disconnect(struct plugin_handle* plugin, struct plugin_user* user) { struct hub_user* huser = convert_user_type(plugin, user); if(huser != NULL) hub_disconnect_user(plugin_get_hub(plugin), huser, quit_kicked); return 0; } static int cbfunc_command_add(struct plugin_handle* plugin, struct plugin_command_handle* cmdh) { struct plugin_callback_data* data = get_callback_data(plugin); struct command_handle* command = (struct command_handle*) hub_malloc_zero(sizeof(struct command_handle)); command->prefix = cmdh->prefix; command->length = cmdh->length; command->args = cmdh->args; command->cred = cmdh->cred; command->description = cmdh->description; command->origin = cmdh->origin; command->handler = plugin_command_dispatch; cmdh->internal_handle = command; list_append(data->commands, cmdh); command_add(plugin_get_hub(plugin)->commands, command, (void*) plugin); printf("*** Add plugin command: %s (%p, %p)\n", command->prefix, command, cmdh); return 0; } static int cbfunc_command_del(struct plugin_handle* plugin, struct plugin_command_handle* cmdh) { struct plugin_callback_data* data = get_callback_data(plugin); struct command_handle* command = (struct command_handle*) cmdh->internal_handle; printf("*** Del plugin command: %s (%p, %p)\n", command->prefix, command, cmdh); list_remove(data->commands, cmdh); command_del(plugin_get_hub(plugin)->commands, command); hub_free(command); cmdh->internal_handle = NULL; return 0; } size_t cbfunc_command_arg_reset(struct plugin_handle* plugin, struct plugin_command* cmd) { // TODO: Use proper function for rewriting for plugin_command -> hub_command return hub_command_arg_reset((struct hub_command*) cmd); } struct plugin_command_arg_data* cbfunc_command_arg_next(struct plugin_handle* plugin, struct plugin_command* cmd, enum plugin_command_arg_type t) { // TODO: Use proper function for rewriting for plugin_command -> hub_command return (struct plugin_command_arg_data*) hub_command_arg_next((struct hub_command*) cmd, (enum hub_command_arg_type) t); } static char* cbfunc_get_hub_name(struct plugin_handle* plugin) { struct hub_info* hub = plugin_get_hub(plugin); char* str_encoded = adc_msg_get_named_argument(hub->command_info, ADC_INF_FLAG_NICK); char* str = adc_msg_unescape(str_encoded); hub_free(str_encoded); return str; } static char* cbfunc_get_hub_description(struct plugin_handle* plugin) { struct hub_info* hub = plugin_get_hub(plugin); char* str_encoded = adc_msg_get_named_argument(hub->command_info, ADC_INF_FLAG_DESCRIPTION); char* str = adc_msg_unescape(str_encoded); hub_free(str_encoded); return str; } static void cbfunc_set_hub_name(struct plugin_handle* plugin, const char* str) { struct hub_info* hub = plugin_get_hub(plugin); struct adc_message* command; char* new_str = adc_msg_escape(str ? str : hub->config->hub_name); adc_msg_replace_named_argument(hub->command_info, ADC_INF_FLAG_NICK, new_str); // Broadcast hub name command = adc_msg_construct(ADC_CMD_IINF, (strlen(new_str) + 8)); adc_msg_add_named_argument(command, ADC_INF_FLAG_NICK, new_str); route_to_all(hub, command); adc_msg_free(command); hub_free(new_str); } static void cbfunc_set_hub_description(struct plugin_handle* plugin, const char* str) { struct hub_info* hub = plugin_get_hub(plugin); struct adc_message* command; char* new_str = adc_msg_escape(str ? str : hub->config->hub_description); adc_msg_replace_named_argument(hub->command_info, ADC_INF_FLAG_DESCRIPTION, new_str); // Broadcast hub description command = adc_msg_construct(ADC_CMD_IINF, (strlen(new_str) + 8)); adc_msg_add_named_argument(command, ADC_INF_FLAG_DESCRIPTION, new_str); route_to_all(hub, command); adc_msg_free(command); hub_free(new_str); } /* Get a list of users currently connected to the hub. The list can be filtered * with the credentials parameter: * - auth_cred_none means no filtering i.e., everybody returned. * - Any of the other auth_cred_xxx values means only users of that credential * level are returned. * - The negative of an auth_cred_xxx value means only users of at least that * credential level are returned. For example, -auth_cred_operators returns * any operators or admins. * * NULL is returned on error, and an empty list is returned if no users match * the requested credentials. */ static struct linked_list* cbfunc_get_user_list(struct plugin_handle* plugin, enum auth_credentials credentials) { /* Determine the comparison mode. */ int atleast = 0; if((int)credentials < 0) { credentials = -credentials; atleast = 1; } /* Check the credential level is valid. */ uhub_assert(credentials <= auth_cred_admin); /* Get the master user list and prepare our copy. */ struct hub_info* hub = plugin_get_hub(plugin); struct linked_list* orig_list = hub->users->list; struct linked_list* new_list = list_create(); if(new_list == NULL) { plugin->error_msg = "Unable to allocate memory for user list"; return NULL; } /* Go through each connected user. */ struct hub_user* user = (struct hub_user*)list_get_first(orig_list); while(user != NULL) { /* Check if we should be including them in the output. */ int include = 0; if(credentials == 0) include = 1; else { if(atleast) { if(user->credentials >= credentials) include = 1; } else { if(user->credentials == credentials) include = 1; } } /* Do we need to include this user? */ if(include) { /* Try to allocate space. We are going to make a copy of the user * data in case the user disconnects before the plugin uses the * list. This way, any hub functions the plugin tries to call will * fail, but at least it won't be trying to access free'd memory. */ struct plugin_user* puser = (struct plugin_user*)hub_malloc(sizeof(struct plugin_user)); if(puser == NULL) { plugin->error_msg = "Unable to allocate memory for list entry in get_user_list."; list_clear(new_list, &hub_free); list_destroy(new_list); return NULL; } /* Copy the pertinent information across and add it to the list. */ memcpy(puser, user, sizeof(struct plugin_user)); list_append(new_list, puser); } /* Next user please. */ user = (struct hub_user*)list_get_next(orig_list); } /* Done. */ return new_list; } /* Clean up the memory used by a user list. */ static void cbfunc_free_user_list(struct plugin_handle* handle, struct linked_list* list) { if(list != NULL) { list_clear(list, &hub_free); list_destroy(list); } } void plugin_register_callback_functions(struct plugin_handle* handle) { handle->hub.send_message = cbfunc_send_message; handle->hub.send_status_message = cbfunc_send_status; handle->hub.user_disconnect = cbfunc_user_disconnect; handle->hub.command_add = cbfunc_command_add; handle->hub.command_del = cbfunc_command_del; handle->hub.command_arg_reset = cbfunc_command_arg_reset; handle->hub.command_arg_next = cbfunc_command_arg_next; handle->hub.get_name = cbfunc_get_hub_name; handle->hub.set_name = cbfunc_set_hub_name; handle->hub.get_description = cbfunc_get_hub_description; handle->hub.set_description = cbfunc_set_hub_description; handle->hub.ucmd_create = cbfunc_ucmd_create; handle->hub.ucmd_add_chat = cbfunc_ucmd_add_chat; handle->hub.ucmd_add_pm = cbfunc_ucmd_add_pm; handle->hub.ucmd_send = cbfunc_ucmd_send; handle->hub.ucmd_free = cbfunc_ucmd_free; handle->hub.get_user_list = cbfunc_get_user_list; handle->hub.free_user_list = cbfunc_free_user_list; } void plugin_unregister_callback_functions(struct plugin_handle* handle) { } struct plugin_callback_data* plugin_callback_data_create() { struct plugin_callback_data* data = (struct plugin_callback_data*) hub_malloc_zero(sizeof(struct plugin_callback_data)); LOG_PLUGIN("plugin_callback_data_create()"); data->commands = list_create(); return data; } void plugin_callback_data_destroy(struct plugin_handle* plugin, struct plugin_callback_data* data) { LOG_PLUGIN("plugin_callback_data_destroy()"); if (data->commands) { // delete commands not deleted by the plugin itself: struct plugin_command_handle* cmd; while ( (cmd = list_get_first(data->commands)) ) cbfunc_command_del(plugin, cmd); list_destroy(data->commands); } hub_free(data); }