diff --git a/GNUmakefile b/GNUmakefile index abc2db2..d48d663 100644 --- a/GNUmakefile +++ b/GNUmakefile @@ -152,6 +152,7 @@ libuhub_SOURCES := \ src/core/plugincallback.c \ src/core/plugininvoke.c \ src/core/pluginloader.c \ + src/core/pluginucmd.c \ src/network/backend.c \ src/network/connection.c \ src/network/epoll.c \ @@ -236,6 +237,9 @@ plugin_chat_only_TARGET := mod_chat_only.so plugin_topic_SOURCES := src/plugins/mod_topic.c plugin_topic_TARGET := mod_topic.so +plugin_ucmd_SOURCES := src/plugins/mod_ucmd.c +plugin_ucmd_TARGET := mod_ucmd.so + # Source to objects libuhub_OBJECTS := $(libuhub_SOURCES:.c=.o) libutils_OBJECTS := $(libutils_SOURCES:.c=.o) @@ -255,7 +259,8 @@ all_plugins := \ $(plugin_welcome_TARGET) \ $(plugin_chat_history_TARGET) \ $(plugin_chat_only_TARGET) \ - $(plugin_topic_TARGET) + $(plugin_topic_TARGET) \ + $(plugin_ucmd_TARGET) all_OBJECTS := \ $(libuhub_OBJECTS) \ @@ -314,6 +319,8 @@ $(plugin_welcome_TARGET): $(plugin_welcome_SOURCES) $(libutils_OBJECTS) $(plugin_topic_TARGET): $(plugin_topic_SOURCES) $(libutils_OBJECTS) $(MSG_CC) $(CC) -shared -fPIC -o $@ $^ $(CFLAGS) +$(plugin_ucmd_TARGET): $(plugin_ucmd_SOURCES) $(libutils_OBJECTS) + $(MSG_CC) $(CC) -shared -fPIC -o $@ $^ $(CFLAGS) $(adcrush_BINARY): $(adcrush_OBJECTS) $(libuhub_OBJECTS) $(libutils_OBJECTS) $(libadc_common_OBJECTS) $(libadc_client_OBJECTS) $(MSG_LD) $(CC) -o $@ $^ $(LDFLAGS) $(LDLIBS) @@ -364,6 +371,7 @@ install: all @if [ ! -f $(UHUB_CONF_DIR)/uhub.conf ]; then cp doc/uhub.conf $(UHUB_CONF_DIR); fi @if [ ! -f $(UHUB_CONF_DIR)/rules.txt ]; then cp doc/rules.txt $(UHUB_CONF_DIR); fi @if [ ! -f $(UHUB_CONF_DIR)/plugins.conf ]; then cp doc/plugins.conf $(UHUB_CONF_DIR); fi + @if [ ! -f $(UHUB_CONF_DIR)/ucmd.conf ]; then cp doc/ucmd.conf $(UHUB_CONF_DIR); fi @if [ ! -d $(UHUB_MOD_DIR) ]; then echo Creating $(UHUB_MOD_DIR); mkdir -p $(UHUB_MOD_DIR); fi @cp -f mod_*.so $(UHUB_MOD_DIR) @touch $(UHUB_CONF_DIR)/motd.txt diff --git a/doc/plugins.conf b/doc/plugins.conf index e45f35c..ff15da7 100644 --- a/doc/plugins.conf +++ b/doc/plugins.conf @@ -60,3 +60,15 @@ plugin /var/lib/uhub/mod_welcome.so "motd=/etc/uhub/motd.txt rules=/etc/uhub/rul # history_connect: the number of chat history messages to send when users connect (0 = do not send any history) plugin /var/lib/uhub/mod_chat_history.so "history_max=200 history_default=10 history_connect=5" +# A plugin to send user commands. These are custom commands which are shown in +# the right-click menu of the users client software and allows easy usage of +# hub commands. +# +# Parameters: +# file: the file where the list of user commands is stored. +# +# NB. Any changes to the commands (if you either change the file location, or +# edit the file itself) will be picked up when you reload the configuration. +# However, the menu that users see is not auto-refreshed i.e., users will not +# see the changes to the commands until they disconnect and reconnect. +plugin /var/lib/uhub/mod_ucmd.so "file=/etc/uhub/ucmd.conf" diff --git a/doc/ucmd.conf b/doc/ucmd.conf new file mode 100644 index 0000000..1289a39 --- /dev/null +++ b/doc/ucmd.conf @@ -0,0 +1,168 @@ +# User command configuration for standard uhub commands. +# +# Each command contains at least two lines: the first defines the command name +# and required credentials, and the following lines define the action(s) taken +# by the client when the user selects the command. +# +# Leading and trailing whitespace is trimmed from each line. Commented lines +# and blank lines are ignored. All keywords in the file are not case sensitive. +# +# Command definition +# ------------------ +# +# The first line of a command defines the credentials required, the context(s) +# that the command should be shown in, and the name it is displayed as in the +# user command menu. It takes the following format: +# +# +# +# The credential level is the minimum credentials required to access the +# command. Anybody with this level or greater will be able to see it. The +# levels are: 0 = none, 1 = bot, 2 = unregistered user, 3 = registered user, +# 4 = operator, 5 = super (not used currently), 6 = link (not used currently), +# 7 = admin. +# +# The contexts are a comma separated list which tell the client where to +# display the command. Possible values are: +# +# * hub: in a general right-click menu for the hub +# * user: in a right-click menu in the user list +# * search: in a search results window +# * filelist: in a file list window +# * all: all of the above +# +# The command name is what is displayed in the client menu and *must* be unique +# as the clients use it as an identifier. +# +# Nested menus can be created by adding backslashes (\) to the name, for example, +# User actions\Kick. NB. the ADC UCMD extension specifies a forward slash (/), +# but all clients appear to use the backslash as per the old NMDC protocol, so +# the backslash is recommended. +# +# Actions +# ------- +# +# Following the command definition, one or more lines are given to specify the +# action or actions the client should take when the user selects the command. +# The order of the actions in this file is the order the client will perform +# them. There are three actions available, each of which can be used multiple +# times in a command: +# +# Chat +# Sends a message in the main chat window. The parameter is 0 or 1, with 1 +# meaning it is formatted as a /me style message by clients. +# +# PM +# Sends a private message. If you have set up a user with a reserved SID, you +# can specify the SID the target. Alternatively, you can use the word Selected +# as the target to specify the currently selected user (this won't work in the +# hub context, as there is no user selected there). The parameter can be +# 0 or 1, with 1 specifying the message should be echoed to the sending user as +# well. +# +# Separator +# Specifies that the entry should be displayed as a separator instead of text, +# meaning there is no 'real' action to run. If this is given, any other actions +# will be ignored. A unique name is still required for each separator. +# +# Substitutions +# ------------- +# +# The client can substitute pieces of text into the actions before it sends +# them. The most useful ones are %[myNI], which is replaced with the nickname +# of the user performing the action, and %[userNI], which is replaced with the +# nickname of the user that was selected in a user list when the command was +# run. The %[userNI] substitution does not work in the hub context, as there is +# no user selected there. +# +# You can also prompt the user for a piece of text with the substitution +# %[line:]. If the same prompt is used multiple times +# within a command (whether in the same or different actions), the user is only +# asked once and the response is used for all instances of the prompt. +# +# For a full list of available substitutions and the contexts they work in, see +# the specification for the UCMD extension, currently available at +# http://adc.sourceforge.net/versions/ADC-EXT-1.0.6.html#_ucmd_user_commands + + +# Basic commands available to all users. +# These don't strictly belong in the user context, but it is often a good idea +# to put them there because the user list is a common place to right-click and +# people probably expect to see them there. +0 hub,user Show my IP +Chat 0 !myip +0 hub,user Hub uptime +Chat 0 !uptime +0 hub,user Hub version +Chat 0 !version +0 hub,user BasicSeparator +Separator +0 hub,user Message of the day +Chat 0 !motd +0 hub,user Hub rules +Chat 0 !rules +0 hub,user Chat history +Chat 0 !history %[line:How many lines of history do you want to see?] + +# Put a separator before operator commands. +4 user OpSeparator +Separator + +# Kick and user info commands. +4 user Kick user +PM 0 Selected You are being kicked: %[line:Reason for kick] +Chat 1 is kicking %[userNI]: %[line:Reason for kick] +Chat 0 !kick %[userNI] +4 user Get user's IP +Chat 0 !getip %[userNI] +4 user,hub Find user with certain IP +Chat 0 !whoip %[line:Enter IP] +4 user,hub Find users within IP range +Chat 0 !whoip %[line:Enter range (can use - or / CIDR notation)] + +# Log/broadcast commands. +4 user,hub LogSeparator +Separator +4 user,hub Show log +Chat 0 !log +4 user,hub Broadcast PM to all users +Chat 0 !broadcast %[line:Enter message to broadcast to all users] + +# Admin commands. +7 user,hub AdminSeparator +Separator +7 user,hub Hub statistics +Chat 0 !stats +7 user,hub Reload configuration files +Chat 0 !reload +7 user,hub Shutdown hub +Chat 0 !shutdown + +# Finish off with the help command after a separator. +0 hub,user HelpSeparator +Separator +0 hub,user Help +Chat 0 !help + +# Sample configuration for a quote recording bot connected to the hub, with +# commands displayed in a sub-menu. +# +# This assumes you have reserved_sids=QuoteBot in your uhub configuration file, +# which will reserve the SID AAAB for the QuoteBot user (remember you must also +# register an account for QuoteBot for this to work). Note that if the bot is +# offline the hub will drop any message destined for the bot. +# +# NB. if you just want to see how this looks, you can uncomment the following +# lines even if you don't have such a bot - the commands will still be sent out +# to users, they just won't do anything. +# +#0 hub,user QuoteBot\Get latest quote +#PM 0 AAAB !latest +#0 hub,user QuoteBot\Get random quote +#PM 0 AAAB !random +#0 hub,user QuoteBot\Separator +#Separator +#0 hub,user QuoteBot\Add quote +#PM 0 AAAB !add %[line:Enter quote to add] +#4 hub,user QuoteBot\Delete quote +#PM 0 AAAB !delete %[line:Enter number of quote to delete] diff --git a/doc/uhub.conf b/doc/uhub.conf index d379b0c..b6430c4 100644 --- a/doc/uhub.conf +++ b/doc/uhub.conf @@ -28,6 +28,13 @@ show_banner_sys_info=1 # Allow only registered users on the hub if set to 1. registered_users_only=0 +# Reserve SIDs for certain users. These users must have a registered account. +# CHANGES TO THIS VALUE WILL NOT BE NOTICED BY A CONFIGURATION RELOAD, YOU +# MUST RESTART THE HUB FOR THEM TO TAKE EFFECT. +# +# QuoteBot = AAAB, StatBot = AAAC +# reserved_sids=QuoteBot StatBot + # A server name and description. hub_name=my hub hub_description=Powered by uHub diff --git a/src/core/commands.c b/src/core/commands.c index fb3256c..4bb1401 100644 --- a/src/core/commands.c +++ b/src/core/commands.c @@ -450,7 +450,7 @@ static int command_whoip(struct command_base* cbase, struct hub_user* user, stru static int command_broadcast(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd) { struct hub_command_arg_data* arg = hub_command_arg_next(cmd, type_string); - char* message = arg->data.string; + char* message = adc_msg_escape(arg->data.string); size_t message_len = strlen(message); char pm_flag[7] = "PM"; char from_sid[5]; @@ -485,6 +485,7 @@ static int command_broadcast(struct command_base* cbase, struct hub_user* user, cbuf_append_format(buf, "*** %s: Delivered to " PRINTF_SIZE_T " user%s", cmd->prefix, recipients, (recipients != 1 ? "s" : "")); send_message(cbase, user, buf); + hub_free(message); return 0; } diff --git a/src/core/config.xml b/src/core/config.xml index b5e949d..11b2635 100644 --- a/src/core/config.xml +++ b/src/core/config.xml @@ -117,6 +117,27 @@ 0.4.0 + +