2020-06-30 22:03:05 +00:00

183 lines
6.5 KiB
Python

###
# Copyright (c) 2020, oddluck <oddluck@riseup.net>
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright notice,
# this list of conditions, and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions, and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of the author of this software nor the name of
# contributors to this software may be used to endorse or promote products
# derived from this software without specific prior written consent.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
###
import supybot.utils as utils
from supybot.commands import *
import supybot.plugins as plugins
import supybot.ircutils as ircutils
import supybot.callbacks as callbacks
import supybot.ircmsgs as ircmsgs
import os
import pexpect
import re
try:
from supybot.i18n import PluginInternationalization
_ = PluginInternationalization("TextAdventures")
except ImportError:
# Placeholder that allows to run the plugin on a bot
# without the i18n module
_ = lambda x: x
class TextAdventures(callbacks.Plugin):
"""
Play Text Adventure Games (Infocom, Interactive Fiction, Z-Machine) .
"""
threaded = True
def __init__(self, irc):
self.__parent = super(TextAdventures, self)
self.__parent.__init__(irc)
self.game = {}
self.game_path = "{0}/games/".format(os.path.dirname(os.path.abspath(__file__)))
self.binary = self.registryValue("dFrotzPath")
def adventure(self, irc, msg, args, input):
"""<game_name>
Open <game_name.z*>.
"""
channel = msg.args[0]
if not self.registryValue("allowPrivate") and not irc.isChannel(channel):
irc.reply("Sorry, this game must be played in channel")
return
elif self.registryValue("allowPrivate") and not irc.isChannel(channel):
channel = msg.nick
game_name = input
self.game.setdefault(channel, None)
if self.game[channel]:
irc.reply(
"There is a game already in progress on {0}. Please stop that game"
" first.".format(channel)
)
else:
irc.reply("Starting {0} on {1}. Please wait...".format(game_name, channel))
game_file = "{0}{1}".format(self.game_path, game_name)
self.game[channel] = pexpect.spawn(
"{0} -m -S 0 {1}".format(self.binary, game_file)
)
response = self.output(self.game[channel])
for line in response:
if line.strip() and line.strip() != ".":
irc.reply(line, prefixNick=False)
adventure = wrap(adventure, ["text"])
def output(self, output):
response = []
prompts = ["\n> >", "\n>", "\n\)", pexpect.TIMEOUT]
output.expect(prompts, timeout=1)
response = output.before
response = response.decode().splitlines()
return response
def doPrivmsg(self, irc, msg):
channel = msg.args[0]
if irc.isChannel(channel) and callbacks.addressed(irc.nick, msg):
return
if not irc.isChannel(channel):
channel = msg.nick
if not self.registryValue("requireCommand") or not irc.isChannel(msg.args[0]):
self.game.setdefault(channel, None)
if self.game[channel]:
try:
command = msg.args[1]
self.game[channel].sendline(r"{}".format(command))
response = self.output(self.game[channel])
for line in response[1:]:
if line.strip() and line.strip() != ".":
irc.reply(line, prefixNick=False)
except:
return
else:
return
else:
return
def end(self, irc, msg, args):
"""
End text adventure game.
"""
channel = msg.args[0]
if not irc.isChannel(channel):
channel = msg.nick
self.game.setdefault(channel, None)
if self.game[channel]:
irc.reply("Stopping Game. Thanks for playing.")
else:
irc.reply("No game running in {0}".format(channel))
try:
self.game[channel].terminate(force=True)
del self.game[channel]
except:
try:
del self.game[channel]
except:
return
return
end = wrap(end)
def z(self, irc, msg, args, command):
"""[<input>]
Send user input or blank line (ENTER/RETURN) to the game.
"""
channel = msg.args[0]
if not irc.isChannel(channel):
channel = msg.nick
self.game.setdefault(channel, None)
if self.game[channel]:
if command:
command = re.sub(r"^.?z", r"", r"{}".format(msg.args[1])).strip()
self.game[channel].sendline(r"{}".format(command))
else:
self.game[channel].sendline()
response = self.output(self.game[channel])
for line in response[1:]:
if line.strip() and line.strip() != ".":
irc.reply(line, prefixNick=False)
else:
irc.reply("No game running in {0}?".format(channel))
z = wrap(z, [optional("text")])
def games(self, irc, msg, args):
"""
List files in the game directory.
"""
reply = ", ".join(sorted(os.listdir(self.game_path)))
irc.reply(reply, prefixNick=False)
games = wrap(games)
Class = TextAdventures