Created
January 13, 2022 04:24
-
-
Save tmarki/b4b185db7883a2f14a8580be2f5bff4b to your computer and use it in GitHub Desktop.
IRC Uno bot for Phenny
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
""" | |
Copyright 2010 Tamas Marki. All rights reserved. | |
Redistribution and use in source and binary forms, with or without modification, are | |
permitted provided that the following conditions are met: | |
1. Redistributions of source code must retain the above copyright notice, this list of | |
conditions and the following disclaimer. | |
2. 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. | |
THIS SOFTWARE IS PROVIDED BY TAMAS MARKI ``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 TAMAS MARKI 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. | |
[18:03] <Lako> .play w r | |
[18:03] <unobot> TopMobil's turn. Top Card: [*] | |
[18:03] [Notice] -unobot- Your cards: [4][9][4][8][D2][D2] | |
[18:03] [Notice] -unobot- Next: hatcher (5 cards) - Lako (2 cards) | |
[18:03] <TopMobil> :O | |
[18:03] <Lako> :O | |
""" | |
import random | |
from datetime import datetime, timedelta | |
CHANNEL = '#unotest' | |
SCOREFILE = "unoscores.txt" | |
STRINGS = { | |
'ALREADY_STARTED' : '\x0300,01Game already started by %s! Type join to join!', | |
'GAME_STARTED' : '\x0300,01IRC-UNO started by %s - Type join to join!', | |
'GAME_STOPPED' : '\x0300,01Game stopped.', | |
'CANT_STOP' : '\x0300,01%s is the game owner, you can\'t stop it!', | |
'DEALING_IN' : '\x0300,01Dealing %s into the game as player #%s!', | |
'JOINED' : '\x0300,01Dealing %s into the game as player #%s!', | |
'ENOUGH' : '\x0300,01There are enough players, type .deal to start!', | |
'NOT_STARTED' : '\x0300,01Game not started, type .uno to start!', | |
'NOT_ENOUGH' : '\x0300,01Not enough players to deal yet.', | |
'NEEDS_TO_DEAL' : '\x0300,01%s needs to deal.', | |
'ALREADY_DEALT' : '\x0300,01Already dealt.', | |
'ON_TURN' : '\x0300,01It\'s %s\'s turn.', | |
'DONT_HAVE' : '\x0300,01You don\'t have that card, %s', | |
'DOESNT_PLAY' : '\x0300,01That card does not play, %s', | |
'UNO' : '\x0300,01UNO! %s has ONE card left!', | |
'WIN' : '\x0300,01We have a winner! %s!!!! This game took %s', | |
'DRAWN_ALREADY' : '\x0300,01You\'ve already drawn, either .pass or .play!', | |
'DRAWS' : '\x0300,01%s draws a card', | |
'DRAWN_CARD' : '\x0300,01Drawn card: %s', | |
'DRAW_FIRST' : '\x0300,01%s, you need to draw first!', | |
'PASSED' : '\x0300,01%s passed!', | |
'NO_SCORES' : '\x0300,01No scores yet', | |
'SCORE_ROW' : '\x0300,01#%s %s (%s points %s games, %s won, %s wasted)', | |
'TOP_CARD' : '\x0300,01%s\'s turn. Top Card: %s', | |
'YOUR_CARDS' : '\x0300,01Your cards: %s', | |
'NEXT_START' : '\x0300,01Next: ', | |
'NEXT_PLAYER' : '\x0300,01%s (%s cards)', | |
'D2' : '\x0300,01%s draws two and is skipped!', | |
'CARDS' : '\x0300,01Cards: %s', | |
'WD4' : '\x0300,01%s draws four and is skipped!', | |
'SKIPPED' : '\x0300,01%s is skipped!', | |
'REVERSED' : '\x0300,01Order reversed!', | |
'GAINS' : '\x0300,01%s gains %s points!', | |
} | |
class UnoBot: | |
def __init__ (self): | |
self.colored_card_nums = [ '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'R', 'S', 'D2' ] | |
self.special_scores = { 'R' : 20, 'S' : 20, 'D2' : 20, 'WD4' : 50, 'W' : 50 } | |
self.colors = 'RGBY' | |
self.special_cards = [ 'W', 'WD4' ] | |
self.players = { } | |
self.playerOrder = [ ] | |
self.game_on = False | |
self.currentPlayer = 0 | |
self.topCard = None | |
self.way = 1 | |
self.drawn = False | |
self.scoreFile = SCOREFILE | |
self.deck = [ ] | |
def start(self, phenny, owner): | |
if self.game_on: | |
phenny.msg (CHANNEL, STRINGS['ALREADY_STARTED'] % self.game_on) | |
else: | |
self.game_on = owner | |
self.deck = [ ] | |
phenny.msg (CHANNEL, STRINGS['GAME_STARTED'] % owner) | |
self.players = { } | |
self.players[owner] = [ ] | |
self.playerOrder = [ owner ] | |
def stop (self, phenny, input): | |
if input.nick == self.game_on: | |
phenny.msg (CHANNEL, STRINGS['GAME_STOPPED']) | |
self.game_on = False | |
elif self.game_on: | |
phenny.msg (CHANNEL, STRINGS['CANT_STOP'] % self.game_on) | |
def join (self, phenny, input): | |
#print dir (phenny.bot) | |
#print dir (input) | |
if self.game_on: | |
if input.nick not in self.players: | |
self.players[input.nick] = [ ] | |
self.playerOrder.append (input.nick) | |
if self.deck: | |
for i in xrange (0, 7): | |
self.players[input.nick].append (self.getCard ()) | |
phenny.msg (CHANNEL, STRINGS['DEALING_IN'] % (input.nick, self.playerOrder.index (input.nick) + 1)) | |
else: | |
phenny.msg (CHANNEL, STRINGS['JOINED'] % (input.nick, self.playerOrder.index (input.nick) + 1)) | |
if len (self.players) > 1: | |
phenny.msg (CHANNEL, STRINGS['ENOUGH']) | |
else: | |
phenny.msg (CHANNEL, STRINGS['NOT_STARTED']) | |
def deal (self, phenny, input): | |
if not self.game_on: | |
phenny.msg (CHANNEL, STRINGS['NOT_STARTED']) | |
return | |
if len (self.players) < 2: | |
phenny.msg (CHANNEL, STRINGS['NOT_ENOUGH']) | |
return | |
if input.nick != self.game_on: | |
phenny.msg (CHANNEL, STRINGS['NEEDS_TO_DEAL'] % self.game_on) | |
return | |
if len (self.deck): | |
phenny.msg (CHANNEL, STRINGS['ALREADY_DEALT']) | |
return | |
self.startTime = datetime.now () | |
self.deck = self.createnewdeck () | |
for i in xrange (0, 7): | |
for p in self.players: | |
self.players[p].append (self.getCard ()) | |
self.topCard = self.getCard () | |
while self.topCard in ['W', 'WD4']: self.topCard = self.getCard () | |
self.currentPlayer = 1 | |
self.cardPlayed (phenny, self.topCard) | |
self.showOnTurn (phenny) | |
def play (self, phenny, input): | |
if not self.game_on or not self.deck: | |
return | |
if input.nick != self.playerOrder[self.currentPlayer]: | |
phenny.msg (CHANNEL, STRINGS['ON_TURN'] % self.playerOrder[self.currentPlayer]) | |
return | |
tok = [z.strip () for z in str (input).upper ().split (' ')] | |
if len (tok) != 3: | |
return | |
searchcard = '' | |
if tok[1] in self.special_cards: | |
searchcard = tok[1] | |
else: searchcard = (tok[1] + tok[2]) | |
if searchcard not in self.players[self.playerOrder[self.currentPlayer]]: | |
phenny.msg (CHANNEL, STRINGS['DONT_HAVE'] % self.playerOrder[self.currentPlayer]) | |
return | |
playcard = (tok[1] + tok[2]) | |
if not self.cardPlayable (playcard): | |
phenny.msg (CHANNEL, STRINGS['DOESNT_PLAY'] % self.playerOrder[self.currentPlayer]) | |
return | |
self.drawn = False | |
self.players[self.playerOrder[self.currentPlayer]].remove (searchcard) | |
pl = self.currentPlayer | |
self.incPlayer () | |
self.cardPlayed (phenny, playcard) | |
if len (self.players[self.playerOrder[pl]]) == 1: | |
phenny.msg (CHANNEL, STRINGS['UNO'] % self.playerOrder[pl]) | |
elif len (self.players[self.playerOrder[pl]]) == 0: | |
phenny.msg (CHANNEL, STRINGS['WIN'] % (self.playerOrder[pl], (datetime.now () - self.startTime))) | |
self.gameEnded (phenny, self.playerOrder[pl]) | |
return | |
self.showOnTurn (phenny) | |
def draw (self, phenny, input): | |
if not self.game_on or not self.deck: | |
return | |
if input.nick != self.playerOrder[self.currentPlayer]: | |
phenny.msg (CHANNEL, STRINGS['ON_TURN'] % self.playerOrder[self.currentPlayer]) | |
return | |
if self.drawn: | |
phenny.msg (CHANNEL, STRINGS['DRAWN_ALREADY']) | |
return | |
self.drawn = True | |
phenny.msg (CHANNEL, STRINGS['DRAWS'] % self.playerOrder[self.currentPlayer]) | |
c = self.getCard () | |
self.players[self.playerOrder[self.currentPlayer]].append (c) | |
phenny.notice (input.nick, STRINGS['DRAWN_CARD'] % self.renderCards ([c])) | |
# this is not a typo, avoiding collision with Python's pass keyword | |
def passs (self, phenny, input): | |
if not self.game_on or not self.deck: | |
return | |
if input.nick != self.playerOrder[self.currentPlayer]: | |
phenny.msg (CHANNEL, STRINGS['ON_TURN'] % self.playerOrder[self.currentPlayer]) | |
return | |
if not self.drawn: | |
phenny.msg (CHANNEL, STRINGS['DRAW_FIRST'] % self.playerOrder[self.currentPlayer]) | |
return | |
self.drawn = False | |
phenny.msg (CHANNEL, STRINGS['PASSED'] % self.playerOrder[self.currentPlayer]) | |
self.incPlayer () | |
self.showOnTurn (phenny) | |
def top10 (self, phenny): | |
from copy import copy | |
prescores = [ ] | |
try: | |
f = open (self.scoreFile, 'r') | |
for l in f: | |
t = l.replace ('\n', '').split (' ') | |
if len (t) < 4: continue | |
prescores.append (copy (t)) | |
if len (t) == 4: t.append (0) | |
f.close () | |
except: pass | |
prescores = sorted (prescores, lambda x, y: cmp ((y[1] != '0') and (float (y[3]) / int (y[1])) or 0, (x[1] != '0') and (float (x[3]) / int (x[1])) or 0)) | |
if not prescores: | |
phenny.msg (CHANNEL, STRINGS['NO_SCORES']) | |
i = 1 | |
for z in prescores[:10]: | |
phenny.msg (CHANNEL, STRINGS['SCORE_ROW'] % (i, z[0], z[3], z[1], z[2], timedelta (seconds = int (z[4])))) | |
i += 1 | |
def createnewdeck (self): | |
ret = [ ] | |
for a in self.colored_card_nums: | |
for b in self.colors: | |
ret.append (b + a) | |
for a in self.special_cards: | |
ret.append (a) | |
ret.append (a) | |
ret *= 2 | |
random.shuffle (ret) | |
return ret | |
def getCard(self): | |
ret = self.deck[0] | |
self.deck.pop (0) | |
if not self.deck: | |
self.deck = self.createnewdeck () | |
return ret | |
def showOnTurn (self, phenny): | |
phenny.msg (CHANNEL, STRINGS['TOP_CARD'] % (self.playerOrder[self.currentPlayer], self.renderCards ([self.topCard]))) | |
phenny.notice (self.playerOrder[self.currentPlayer], STRINGS['YOUR_CARDS'] % self.renderCards (self.players[self.playerOrder[self.currentPlayer]])) | |
msg = STRINGS['NEXT_START'] | |
tmp = self.currentPlayer + self.way | |
if tmp == len (self.players): | |
tmp = 0 | |
if tmp < 0: | |
tmp = len (self.players) - 1 | |
arr = [ ] | |
while tmp != self.currentPlayer: | |
arr.append (STRINGS['NEXT_PLAYER'] % (self.playerOrder[tmp], len (self.players[self.playerOrder[tmp]]))) | |
tmp = tmp + self.way | |
if tmp == len (self.players): | |
tmp = 0 | |
if tmp < 0: | |
tmp = len (self.players) - 1 | |
msg += ' - '.join (arr) | |
phenny.notice (self.playerOrder[self.currentPlayer], msg) | |
def renderCards (self, cards): | |
ret = [ ] | |
for c in sorted (cards): | |
if c in ['W', 'WD4']: | |
ret.append ('\x0300,01[' + c + ']') | |
continue | |
if c[0] == 'W': | |
c = c[-1] + '*' | |
t = '\x0300,01\x03' | |
if c[0] == 'B': | |
t += '11,01[' | |
if c[0] == 'Y': | |
t += '08,01[' | |
if c[0] == 'G': | |
t += '09,01[' | |
if c[0] == 'R': | |
t += '04,01[' | |
t += c[1:] + ']\x0300,01' | |
ret.append (t) | |
return ''.join (ret) | |
def cardPlayable (self, card): | |
if card[0] == 'W' and card[-1] in self.colors: | |
return True | |
if self.topCard[0] == 'W': | |
return card[0] == self.topCard[-1] | |
return ((card[0] == self.topCard[0]) or (card[1] == self.topCard[1])) and (card[0] not in ['W', 'WD4']) | |
def cardPlayed (self, phenny, card): | |
if card[1:] == 'D2': | |
phenny.msg (CHANNEL, STRINGS['D2'] % self.playerOrder[self.currentPlayer]) | |
z = [self.getCard (), self.getCard ()] | |
phenny.notice(self.playerOrder[self.currentPlayer], STRINGS['CARDS'] % self.renderCards (z)) | |
self.players[self.playerOrder[self.currentPlayer]].extend (z) | |
self.incPlayer () | |
elif card[:2] == 'WD': | |
phenny.msg (CHANNEL, STRINGS['WD4'] % self.playerOrder[self.currentPlayer]) | |
z = [self.getCard (), self.getCard (), self.getCard (), self.getCard ()] | |
phenny.notice(self.playerOrder[self.currentPlayer], STRINGS['CARDS'] % self.renderCards (z)) | |
self.players[self.playerOrder[self.currentPlayer]].extend (z) | |
self.incPlayer () | |
elif card[1] == 'S': | |
phenny.msg (CHANNEL, STRINGS['SKIPPED'] % self.playerOrder[self.currentPlayer]) | |
self.incPlayer () | |
elif card[1] == 'R' and card[0] != 'W': | |
phenny.msg (CHANNEL, STRINGS['REVERSED']) | |
self.way = -self.way | |
self.incPlayer () | |
self.incPlayer () | |
self.topCard = card | |
def gameEnded (self, phenny, winner): | |
try: | |
score = 0 | |
for p in self.players: | |
for c in self.players[p]: | |
if c[0] == 'W': | |
score += self.special_scores[c] | |
elif c[1] in [ 'S', 'R', 'D' ]: | |
score += self.special_scores[c[1:]] | |
else: | |
score += int (c[1]) | |
phenny.msg (CHANNEL, STRINGS['GAINS'] % (winner, score)) | |
self.saveScores (self.players.keys (), winner, score, (datetime.now () - self.startTime).seconds) | |
except Exception, e: | |
print 'Score error: %s' % e | |
self.players = { } | |
self.playerOrder = [ ] | |
self.game_on = False | |
self.currentPlayer = 0 | |
self.topCard = None | |
self.way = 1 | |
def incPlayer (self): | |
self.currentPlayer = self.currentPlayer + self.way | |
if self.currentPlayer == len (self.players): | |
self.currentPlayer = 0 | |
if self.currentPlayer < 0: | |
self.currentPlayer = len (self.players) - 1 | |
def saveScores (self, players, winner, score, time): | |
from copy import copy | |
prescores = { } | |
try: | |
f = open (self.scoreFile, 'r') | |
for l in f: | |
t = l.replace ('\n', '').split (' ') | |
if len (t) < 4: continue | |
if len (t) == 4: t.append (0) | |
prescores[t[0]] = [t[0], int (t[1]), int (t[2]), int (t[3]), int (t[4])] | |
f.close () | |
except: pass | |
for p in players: | |
if p not in prescores: | |
prescores[p] = [ p, 0, 0, 0, 0 ] | |
prescores[p][1] += 1 | |
prescores[p][4] += time | |
prescores[winner][2] += 1 | |
prescores[winner][3] += score | |
try: | |
f = open (self.scoreFile, 'w') | |
for p in prescores: | |
f.write (' '.join ([str (s) for s in prescores[p]]) + '\n') | |
f.close () | |
except Exception, e: | |
print 'Failed to write score file %s' % e | |
unobot = UnoBot () | |
def uno(phenny, input): | |
unobot.start (phenny, input.nick) | |
uno.commands = ['uno'] | |
uno.priority = 'high' | |
def unostop(phenny, input): | |
unobot.stop (phenny, input) | |
unostop.commands = ['unostop'] | |
unostop.priority = 'high' | |
def join(phenny, input): | |
unobot.join (phenny, input) | |
join.rule = '^join$' | |
join.priority = 'high' | |
def deal(phenny, input): | |
unobot.deal (phenny, input) | |
deal.commands = ['deal'] | |
deal.priority = 'high' | |
def play(phenny, input): | |
unobot.play (phenny, input) | |
play.commands = ['play'] | |
play.priority = 'high' | |
def draw(phenny, input): | |
unobot.draw (phenny, input) | |
draw.commands = ['draw'] | |
draw.priority = 'high' | |
def passs(phenny, input): | |
unobot.passs (phenny, input) | |
passs.commands = ['pass'] | |
passs.priority = 'high' | |
def unotop10 (phenny, input): | |
unobot.top10 (phenny) | |
unotop10.commands = ['unotop10'] | |
unotop10.priority = 'high' | |
if __name__ == '__main__': | |
print __doc__.strip() | |
For full context: This is an archive to go along with my fork of the original code, updated to work with Willie and then Sopel (successors to Phenny). https://github.com/dgw/sopel-UnoBot
Many thanks to @tmarki for noticing my fork and proactively ensuring that the source work was preserved and linked to it!
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Original (dead) URL: http://tmarki.com/wp/2010/02/18/irc-uno-bot-for-phenny/
Features:
– slightly modified UNO gameplay
– unlimited number of players in a round (minimum 2)
– logging of total score, games played, won, and total time spent playing per nickname
– separated strings, so localization to other languages should be easy
How to set up:
Known issues:
Note that ordinary IRC nicknames are used to identify the players. The bot cannot react to a player leaving the channel or timeouting, either the game must be stopped (if the game owner quit then someone will have to take their nick temporarly to do the .unostop action) or someone must step in with the same nickname.