Source code for evennia.commands.default.help

"""
The help command. The basic idea is that help texts for commands
are best written by those that write the commands - the admins. So
command-help is all auto-loaded and searched from the current command
set. The normal, database-tied help system is used for collaborative
creation of other help topics such as RP help or game-world aides.
"""

from django.conf import settings
from collections import defaultdict
from evennia.utils.utils import fill, dedent
from evennia.commands.command import Command
from evennia.help.models import HelpEntry
from evennia.utils import create, evmore
from evennia.utils.eveditor import EvEditor
from evennia.utils.utils import string_suggestions, class_from_module

COMMAND_DEFAULT_CLASS = class_from_module(settings.COMMAND_DEFAULT_CLASS)
HELP_MORE = settings.HELP_MORE
CMD_IGNORE_PREFIXES = settings.CMD_IGNORE_PREFIXES

# limit symbol import for API
__all__ = ("CmdHelp", "CmdSetHelp")
_DEFAULT_WIDTH = settings.CLIENT_DEFAULT_WIDTH
_SEP = "|C" + "-" * _DEFAULT_WIDTH + "|n"


[docs]class CmdHelp(COMMAND_DEFAULT_CLASS): """ View help or a list of topics Usage: help <topic or command> help list help all This will search for help on commands and other topics related to the game. """ key = "help" aliases = ["?"] locks = "cmd:all()" arg_regex = r"\s|$" # this is a special cmdhandler flag that makes the cmdhandler also pack # the current cmdset with the call to self.func(). return_cmdset = True # Help messages are wrapped in an EvMore call (unless using the webclient # with separate help popups) If you want to avoid this, simply add # 'HELP_MORE = False' in your settings/conf/settings.py help_more = HELP_MORE # suggestion cutoff, between 0 and 1 (1 => perfect match) suggestion_cutoff = 0.6 # number of suggestions (set to 0 to remove suggestions from help) suggestion_maxnum = 5
[docs] def msg_help(self, text): """ messages text to the caller, adding an extra oob argument to indicate that this is a help command result and could be rendered in a separate help window """ if type(self).help_more: usemore = True if self.session and self.session.protocol_key in ("websocket", "ajax/comet"): try: options = self.account.db._saved_webclient_options if options and options["helppopup"]: usemore = False except KeyError: pass if usemore: evmore.msg(self.caller, text, session=self.session) return self.msg(text=(text, {"type": "help"}))
[docs] @staticmethod def format_help_entry(title, help_text, aliases=None, suggested=None): """ This visually formats the help entry. This method can be overridden to customize the way a help entry is displayed. Args: title (str): the title of the help entry. help_text (str): the text of the help entry. aliases (list of str or None): the list of aliases. suggested (list of str or None): suggested reading. Returns the formatted string, ready to be sent. """ string = _SEP + "\n" if title: string += "|CHelp for |w%s|n" % title if aliases: string += " |C(aliases: %s|C)|n" % ("|C,|n ".join("|w%s|n" % ali for ali in aliases)) if help_text: string += "\n%s" % dedent(help_text.rstrip()) if suggested: string += "\n\n|CSuggested:|n " string += "%s" % fill("|C,|n ".join("|w%s|n" % sug for sug in suggested)) string.strip() string += "\n" + _SEP return string
[docs] @staticmethod def format_help_list(hdict_cmds, hdict_db): """ Output a category-ordered list. The input are the pre-loaded help files for commands and database-helpfiles respectively. You can override this method to return a custom display of the list of commands and topics. """ string = "" if hdict_cmds and any(hdict_cmds.values()): string += "\n" + _SEP + "\n |CCommand help entries|n\n" + _SEP for category in sorted(hdict_cmds.keys()): string += "\n |w%s|n:\n" % (str(category).title()) string += "|G" + fill("|C, |G".join(sorted(hdict_cmds[category]))) + "|n" if hdict_db and any(hdict_db.values()): string += "\n\n" + _SEP + "\n\r |COther help entries|n\n" + _SEP for category in sorted(hdict_db.keys()): string += "\n\r |w%s|n:\n" % (str(category).title()) string += ( "|G" + fill(", ".join(sorted([str(topic) for topic in hdict_db[category]]))) + "|n" ) return string
[docs] def check_show_help(self, cmd, caller): """ Helper method. If this return True, the given cmd auto-help will be viewable in the help listing. Override this to easily select what is shown to the account. Note that only commands available in the caller's merged cmdset are available. Args: cmd (Command): Command class from the merged cmdset caller (Character, Account or Session): The current caller executing the help command. """ # return only those with auto_help set and passing the cmd: lock return cmd.auto_help and cmd.access(caller)
[docs] def should_list_cmd(self, cmd, caller): """ Should the specified command appear in the help table? This method only checks whether a specified command should appear in the table of topics/commands. The command can be used by the caller (see the 'check_show_help' method) and the command will still be available, for instance, if a character type 'help name of the command'. However, if you return False, the specified command will not appear in the table. This is sometimes useful to "hide" commands in the table, but still access them through the help system. Args: cmd: the command to be tested. caller: the caller of the help system. Return: True: the command should appear in the table. False: the command shouldn't appear in the table. """ return cmd.access(caller, "view", default=True)
[docs] def parse(self): """ input is a string containing the command or topic to match. """ self.original_args = self.args.strip() self.args = self.args.strip().lower()
[docs] def func(self): """ Run the dynamic help entry creator. """ query, cmdset = self.args, self.cmdset caller = self.caller suggestion_cutoff = self.suggestion_cutoff suggestion_maxnum = self.suggestion_maxnum if not query: query = "all" # removing doublets in cmdset, caused by cmdhandler # having to allow doublet commands to manage exits etc. cmdset.make_unique(caller) # retrieve all available commands and database topics all_cmds = [cmd for cmd in cmdset if self.check_show_help(cmd, caller)] all_topics = [ topic for topic in HelpEntry.objects.all() if topic.access(caller, "view", default=True) ] all_categories = list( set( [cmd.help_category.lower() for cmd in all_cmds] + [topic.help_category.lower() for topic in all_topics] ) ) if query in ("list", "all"): # we want to list all available help entries, grouped by category hdict_cmd = defaultdict(list) hdict_topic = defaultdict(list) # create the dictionaries {category:[topic, topic ...]} required by format_help_list # Filter commands that should be reached by the help # system, but not be displayed in the table, or be displayed differently. for cmd in all_cmds: if self.should_list_cmd(cmd, caller): key = ( cmd.auto_help_display_key if hasattr(cmd, "auto_help_display_key") else cmd.key ) hdict_cmd[cmd.help_category].append(key) [hdict_topic[topic.help_category].append(topic.key) for topic in all_topics] # report back self.msg_help(self.format_help_list(hdict_cmd, hdict_topic)) return # Try to access a particular command # build vocabulary of suggestions and rate them by string similarity. suggestions = None if suggestion_maxnum > 0: vocabulary = ( [cmd.key for cmd in all_cmds if cmd] + [topic.key for topic in all_topics] + all_categories ) [vocabulary.extend(cmd.aliases) for cmd in all_cmds] suggestions = [ sugg for sugg in string_suggestions( query, set(vocabulary), cutoff=suggestion_cutoff, maxnum=suggestion_maxnum ) if sugg != query ] if not suggestions: suggestions = [ sugg for sugg in vocabulary if sugg != query and sugg.startswith(query) ] # try an exact command auto-help match match = [cmd for cmd in all_cmds if cmd == query] if not match: # try an inexact match with prefixes stripped from query and cmds _query = query[1:] if query[0] in CMD_IGNORE_PREFIXES else query match = [ cmd for cmd in all_cmds for m in cmd._matchset if m == _query or m[0] in CMD_IGNORE_PREFIXES and m[1:] == _query ] if len(match) == 1: cmd = match[0] key = cmd.auto_help_display_key if hasattr(cmd, "auto_help_display_key") else cmd.key formatted = self.format_help_entry( key, cmd.get_help(caller, cmdset), aliases=cmd.aliases, suggested=suggestions, ) self.msg_help(formatted) return # try an exact database help entry match match = list(HelpEntry.objects.find_topicmatch(query, exact=True)) if len(match) == 1: formatted = self.format_help_entry( match[0].key, match[0].entrytext, aliases=match[0].aliases.all(), suggested=suggestions, ) self.msg_help(formatted) return # try to see if a category name was entered if query in all_categories: self.msg_help( self.format_help_list( { query: [ cmd.auto_help_display_key if hasattr(cmd, "auto_help_display_key") else cmd.key for cmd in all_cmds if cmd.help_category == query ] }, {query: [topic.key for topic in all_topics if topic.help_category == query]}, ) ) return # no exact matches found. Just give suggestions. self.msg( self.format_help_entry( "", f"No help entry found for '{query}'", None, suggested=suggestions ), options={"type": "help"}, )
def _loadhelp(caller): entry = caller.db._editing_help if entry: return entry.entrytext else: return "" def _savehelp(caller, buffer): entry = caller.db._editing_help caller.msg("Saved help entry.") if entry: entry.entrytext = buffer def _quithelp(caller): caller.msg("Closing the editor.") del caller.db._editing_help
[docs]class CmdSetHelp(COMMAND_DEFAULT_CLASS): """ Edit the help database. Usage: help[/switches] <topic>[[;alias;alias][,category[,locks]] [= <text>] Switches: edit - open a line editor to edit the topic's help text. replace - overwrite existing help topic. append - add text to the end of existing topic with a newline between. extend - as append, but don't add a newline. delete - remove help topic. Examples: sethelp throw = This throws something at ... sethelp/append pickpocketing,Thievery = This steals ... sethelp/replace pickpocketing, ,attr(is_thief) = This steals ... sethelp/edit thievery This command manipulates the help database. A help entry can be created, appended/merged to and deleted. If you don't assign a category, the "General" category will be used. If no lockstring is specified, default is to let everyone read the help file. """ key = "sethelp" switch_options = ("edit", "replace", "append", "extend", "delete") locks = "cmd:perm(Helper)" help_category = "Building"
[docs] def func(self): """Implement the function""" switches = self.switches lhslist = self.lhslist if not self.args: self.msg( "Usage: sethelp[/switches] <topic>[;alias;alias][,category[,locks,..] = <text>" ) return nlist = len(lhslist) topicstr = lhslist[0] if nlist > 0 else "" if not topicstr: self.msg("You have to define a topic!") return topicstrlist = topicstr.split(";") topicstr, aliases = (topicstrlist[0], topicstrlist[1:] if len(topicstr) > 1 else []) aliastxt = ("(aliases: %s)" % ", ".join(aliases)) if aliases else "" old_entry = None # check if we have an old entry with the same name try: for querystr in topicstrlist: old_entry = HelpEntry.objects.find_topicmatch(querystr) # also search by alias if old_entry: old_entry = list(old_entry)[0] break category = lhslist[1] if nlist > 1 else old_entry.help_category lockstring = ",".join(lhslist[2:]) if nlist > 2 else old_entry.locks.get() except Exception: old_entry = None category = lhslist[1] if nlist > 1 else "General" lockstring = ",".join(lhslist[2:]) if nlist > 2 else "view:all()" category = category.lower() if "edit" in switches: # open the line editor to edit the helptext. No = is needed. if old_entry: topicstr = old_entry.key if self.rhs: # we assume append here. old_entry.entrytext += "\n%s" % self.rhs helpentry = old_entry else: helpentry = create.create_help_entry( topicstr, self.rhs, category=category, locks=lockstring, aliases=aliases ) self.caller.db._editing_help = helpentry EvEditor( self.caller, loadfunc=_loadhelp, savefunc=_savehelp, quitfunc=_quithelp, key="topic {}".format(topicstr), persistent=True, ) return if "append" in switches or "merge" in switches or "extend" in switches: # merge/append operations if not old_entry: self.msg("Could not find topic '%s'. You must give an exact name." % topicstr) return if not self.rhs: self.msg("You must supply text to append/merge.") return if "merge" in switches: old_entry.entrytext += " " + self.rhs else: old_entry.entrytext += "\n%s" % self.rhs old_entry.aliases.add(aliases) self.msg("Entry updated:\n%s%s" % (old_entry.entrytext, aliastxt)) return if "delete" in switches or "del" in switches: # delete the help entry if not old_entry: self.msg("Could not find topic '%s'%s." % (topicstr, aliastxt)) return old_entry.delete() self.msg("Deleted help entry '%s'%s." % (topicstr, aliastxt)) return # at this point it means we want to add a new help entry. if not self.rhs: self.msg("You must supply a help text to add.") return if old_entry: if "replace" in switches: # overwrite old entry old_entry.key = topicstr old_entry.entrytext = self.rhs old_entry.help_category = category old_entry.locks.clear() old_entry.locks.add(lockstring) old_entry.aliases.add(aliases) old_entry.save() self.msg("Overwrote the old topic '%s'%s." % (topicstr, aliastxt)) else: self.msg( "Topic '%s'%s already exists. Use /replace to overwrite " "or /append or /merge to add text to it." % (topicstr, aliastxt) ) else: # no old entry. Create a new one. new_entry = create.create_help_entry( topicstr, self.rhs, category=category, locks=lockstring, aliases=aliases ) if new_entry: self.msg("Topic '%s'%s was successfully created." % (topicstr, aliastxt)) if "edit" in switches: # open the line editor to edit the helptext self.caller.db._editing_help = new_entry EvEditor( self.caller, loadfunc=_loadhelp, savefunc=_savehelp, quitfunc=_quithelp, key="topic {}".format(new_entry.key), persistent=True, ) return else: self.msg( "Error when creating topic '%s'%s! Contact an admin." % (topicstr, aliastxt) )