""" Scripts Scripts are powerful jacks-of-all-trades. They have no in-game existence and can be used to represent persistent game systems in some circumstances. Scripts can also have a time component that allows them to "fire" regularly or a limited number of times. There is generally no "tree" of Scripts inheriting from each other. Rather, each script tends to inherit from the base Script class and just overloads its hooks to have it perform its function. """ import random from evennia import DefaultScript, gametime, search_tag, logger from evennia.utils import inherits_from from utils.utils import has_tag, toggle_effect, has_effect class Script(DefaultScript): """ A script type is customized by redefining some or all of its hook methods and variables. * available properties key (string) - name of object name (string)- same as key aliases (list of strings) - aliases to the object. Will be saved to database as AliasDB entries but returned as strings. dbref (int, read-only) - unique #id-number. Also "id" can be used. date_created (string) - time stamp of object creation permissions (list of strings) - list of permission strings desc (string) - optional description of script, shown in listings obj (Object) - optional object that this script is connected to and acts on (set automatically by obj.scripts.add()) interval (int) - how often script should run, in seconds. <0 turns off ticker start_delay (bool) - if the script should start repeating right away or wait self.interval seconds repeats (int) - how many times the script should repeat before stopping. 0 means infinite repeats persistent (bool) - if script should survive a server shutdown or not is_active (bool) - if script is currently running * Handlers locks - lock-handler: use locks.add() to add new lock strings db - attribute-handler: store/retrieve database attributes on this self.db.myattr=val, val=self.db.myattr ndb - non-persistent attribute handler: same as db but does not create a database entry when storing data * Helper methods start() - start script (this usually happens automatically at creation and obj.script.add() etc) stop() - stop script, and delete it pause() - put the script on hold, until unpause() is called. If script is persistent, the pause state will survive a shutdown. unpause() - restart a previously paused script. The script will continue from the paused timer (but at_start() will be called). time_until_next_repeat() - if a timed script (interval>0), returns time until next tick * Hook methods (should also include self as the first argument): at_script_creation() - called only once, when an object of this class is first created. is_valid() - is called to check if the script is valid to be running at the current time. If is_valid() returns False, the running script is stopped and removed from the game. You can use this to check state changes (i.e. an script tracking some combat stats at regular intervals is only valid to run while there is actual combat going on). at_start() - Called every time the script is started, which for persistent scripts is at least once every server start. Note that this is unaffected by self.delay_start, which only delays the first call to at_repeat(). at_repeat() - Called every self.interval seconds. It will be called immediately upon launch unless self.delay_start is True, which will delay the first call of this method by self.interval seconds. If self.interval==0, this method will never be called. at_stop() - Called as the script object is stopped and is about to be removed from the game, e.g. because is_valid() returned False. at_server_reload() - Called when server reloads. Can be used to save temporary variables you want should survive a reload. at_server_shutdown() - called at a full server shutdown. """ pass class CmdActionScript(Script): """ """ def at_script_creation(self): super().at_script_creation() self.start_delay = True self.persistent = True # will survive reload self.repeats = 1 def busy_msg(self): return "doing something else" class AiManagerScript(Script): """ """ def at_script_creation(self): super().at_script_creation() self.key = "ai_manager_script" self.desc = "Does things." def at_start(self): logger.log_info("[AiManagerScript] starting...") bots = search_tag(key="ai_mob", category="general") logger.log_info("[AiManagerScript] found %d ai aware mobs." % len(bots)) logger.log_info("[AiManagerScript] started.") def at_stop(self): logger.log_info("[AiManagerScript] stopped.") def at_repeat(self): current_time = gametime.gametime() bots = search_tag(key="ai_mob", category="general") for bot in bots: bot.think() if bot.db.action != None: bot.db.action.update(bot) if bot.db.action.completion_time() <= current_time: bot.db.action.complete(bot) bot.db.action.delete() class Weather(Script): """ A timer script that displays weather info. Meant to be attached to a room. """ def at_script_creation(self): self.key = "weather_script" self.desc = "Gives random weather messages." self.interval = 60 * 1 # every 5 minutes self.persistent = True # will survive reload def at_repeat(self): "called every self.interval seconds." rand = random.random() if rand < 0.5: weather = "A faint breeze is felt." elif rand < 0.7: weather = "Clouds sweep across the sky." else: weather = "There is a light drizzle of rain." # send this message to everyone inside the object this # script is attached to (likely a room) self.obj.msg_contents(weather)