Party hhc: now it has to work with tournaments
This commit is contained in:
parent
aa62dc47c9
commit
ce43ab4d52
|
@ -1,7 +1,7 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
#
|
#
|
||||||
# Copyright 2008, Carl Gherardi
|
# Copyright 2009, Grigorij Indigirkin
|
||||||
#
|
#
|
||||||
# This program is free software; you can redistribute it and/or modify
|
# This program is free software; you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU General Public License as published by
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -26,23 +26,30 @@ from HandHistoryConverter import *
|
||||||
# PartyPoker HH Format
|
# PartyPoker HH Format
|
||||||
|
|
||||||
class PartyPoker(HandHistoryConverter):
|
class PartyPoker(HandHistoryConverter):
|
||||||
|
class ParsingException(Exception):
|
||||||
|
"Usage: raise ParsingException(<msg>[, hh=<hh>])"
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
if len(args)==0: args=[''] + list(args)
|
||||||
|
msg, args = args[0], args[1:]
|
||||||
|
if 'hh' in kwargs:
|
||||||
|
msg += self.wrapHh(kwargs['hh'])
|
||||||
|
del kwargs['hh']
|
||||||
|
return Exception.__init__(self, msg, *args, **kwargs)
|
||||||
|
def wrapHh(self, hh):
|
||||||
|
return ("\n\nHand history attached below:\n"
|
||||||
|
"%(DELIMETER)s\n%(HH)s\n%(DELIMETER)s") % \
|
||||||
|
{'DELIMETER': '#'*50, 'HH': hh}
|
||||||
|
|
||||||
############################################################
|
############################################################
|
||||||
# Class Variables
|
# Class Variables
|
||||||
|
|
||||||
#mixes = { 'HORSE': 'horse', '8-Game': '8game', 'HOSE': 'hose'} # Legal mixed games
|
|
||||||
sym = {'USD': "\$", }
|
sym = {'USD': "\$", }
|
||||||
#sym = {'USD': "\$", 'CAD': "\$", 'T$': "", "EUR": "\x80", "GBP": "\xa3"} # ADD Euro, Sterling, etc HERE
|
|
||||||
substitutions = {
|
|
||||||
'LEGAL_ISO' : "USD|EUR|GBP|CAD", # legal ISO currency codes
|
|
||||||
'LS' : "\$|\x80|\xa3" # legal currency symbols ADD Euro, Sterling, etc HERE
|
|
||||||
}
|
|
||||||
|
|
||||||
# Static regexes
|
# Static regexes
|
||||||
# $5 USD NL Texas Hold'em - Saturday, July 25, 07:53:52 EDT 2009
|
# $5 USD NL Texas Hold'em - Saturday, July 25, 07:53:52 EDT 2009
|
||||||
# NL Texas Hold'em $1 USD Buy-in Trny:45685440 Level:8 Blinds-Antes(600/1 200 -50) - Sunday, May 17, 11:25:07 MSKS 2009
|
# NL Texas Hold'em $1 USD Buy-in Trny:45685440 Level:8 Blinds-Antes(600/1 200 -50) - Sunday, May 17, 11:25:07 MSKS 2009
|
||||||
re_GameInfoRing = re.compile("""
|
re_GameInfoRing = re.compile("""
|
||||||
(?:\$|)\s*(?P<RINGLIMIT>\d+)\s*(?P<CURRENCY>USD)?\s*
|
(?P<CURRENCY>\$|)\s*(?P<RINGLIMIT>\d+)\s*(?:USD)?\s*
|
||||||
(?P<LIMIT>(NL))\s+
|
(?P<LIMIT>(NL))\s+
|
||||||
(?P<GAME>(Texas\ Hold\'em))
|
(?P<GAME>(Texas\ Hold\'em))
|
||||||
\s*\-\s*
|
\s*\-\s*
|
||||||
|
@ -51,8 +58,7 @@ class PartyPoker(HandHistoryConverter):
|
||||||
re_GameInfoTrny = re.compile("""
|
re_GameInfoTrny = re.compile("""
|
||||||
(?P<LIMIT>(NL))\s+
|
(?P<LIMIT>(NL))\s+
|
||||||
(?P<GAME>(Texas\ Hold\'em))\s+
|
(?P<GAME>(Texas\ Hold\'em))\s+
|
||||||
(?:\$|)\s*
|
(?P<BUYIN>\$?[.0-9]+)\s*(?P<BUYIN_CURRENCY>USD)?\s*Buy-in\s+
|
||||||
(?P<BUYIN>[.0-9]+)\s*(?P<CURRENCY>USD)?\s*Buy-in\s+
|
|
||||||
Trny:\s?(?P<TOURNO>\d+)\s+
|
Trny:\s?(?P<TOURNO>\d+)\s+
|
||||||
Level:\s*(?P<LEVEL>\d+)\s+
|
Level:\s*(?P<LEVEL>\d+)\s+
|
||||||
Blinds(?:-Antes)?\(
|
Blinds(?:-Antes)?\(
|
||||||
|
@ -87,7 +93,7 @@ class PartyPoker(HandHistoryConverter):
|
||||||
re_PlayerInfo = re.compile("""
|
re_PlayerInfo = re.compile("""
|
||||||
Seat\s(?P<SEAT>\d+):\s
|
Seat\s(?P<SEAT>\d+):\s
|
||||||
(?P<PNAME>.*)\s
|
(?P<PNAME>.*)\s
|
||||||
\(\s*\$?(?P<CASH>[0-9,.]+)\s*(?:USD)\s*\)
|
\(\s*\$?(?P<CASH>[0-9,.]+)\s*(?:USD|)\s*\)
|
||||||
""" ,
|
""" ,
|
||||||
re.VERBOSE)
|
re.VERBOSE)
|
||||||
#re_PlayerInfo = re.compile("""
|
#re_PlayerInfo = re.compile("""
|
||||||
|
@ -148,17 +154,19 @@ follow : whether to tail -f the input"""
|
||||||
# They still identify the hero.
|
# They still identify the hero.
|
||||||
self.compiledPlayers = players
|
self.compiledPlayers = players
|
||||||
player_re = "(?P<PNAME>" + "|".join(map(re.escape, players)) + ")"
|
player_re = "(?P<PNAME>" + "|".join(map(re.escape, players)) + ")"
|
||||||
subst = {'PLYR': player_re, 'CUR': hand.gametype['currency']}
|
subst = {'PLYR': player_re, 'CUR_SYM': hand.SYMBOL[hand.gametype['currency']],
|
||||||
|
'CUR': hand.gametype['currency'] if hand.gametype['currency']!='T$' else ''}
|
||||||
logging.debug("player_re: " + subst['PLYR'])
|
logging.debug("player_re: " + subst['PLYR'])
|
||||||
|
logging.debug("CUR_SYM: " + subst['CUR_SYM'])
|
||||||
logging.debug("CUR: " + subst['CUR'])
|
logging.debug("CUR: " + subst['CUR'])
|
||||||
self.re_PostSB = re.compile(
|
self.re_PostSB = re.compile(
|
||||||
r"^%(PLYR)s posts small blind \[[^.0-9]?(?P<SB>[.0-9]+) ?%(CUR)s\]\." % subst,
|
r"^%(PLYR)s posts small blind \[%(CUR_SYM)s(?P<SB>[.0-9]+) ?%(CUR)s\]\." % subst,
|
||||||
re.MULTILINE)
|
re.MULTILINE)
|
||||||
self.re_PostBB = re.compile(
|
self.re_PostBB = re.compile(
|
||||||
r"^%(PLYR)s posts big blind \[[^.0-9]?(?P<BB>[.0-9]+) ?%(CUR)s\]\." % subst,
|
r"^%(PLYR)s posts big blind \[%(CUR_SYM)s(?P<BB>[.0-9]+) ?%(CUR)s\]\." % subst,
|
||||||
re.MULTILINE)
|
re.MULTILINE)
|
||||||
self.re_Antes = re.compile(
|
self.re_Antes = re.compile(
|
||||||
r"^%(PLYR)s posts ante \[[^.,0-9]?(?P<ANTE>[.0-9]+) ?%(CUR)s\]\." % subst,
|
r"^%(PLYR)s posts ante \[%(CUR_SYM)s(?P<ANTE>[.0-9]+) ?%(CUR)s\]\." % subst,
|
||||||
re.MULTILINE)
|
re.MULTILINE)
|
||||||
#self.re_BringIn = re.compile(
|
#self.re_BringIn = re.compile(
|
||||||
#r"^%(PLYR)s: brings[- ]in( low|) for %(CUR)s(?P<BRINGIN>[.0-9]+)" % subst,
|
#r"^%(PLYR)s: brings[- ]in( low|) for %(CUR)s(?P<BRINGIN>[.0-9]+)" % subst,
|
||||||
|
@ -171,7 +179,7 @@ follow : whether to tail -f the input"""
|
||||||
re.MULTILINE)
|
re.MULTILINE)
|
||||||
self.re_Action = re.compile(r"""
|
self.re_Action = re.compile(r"""
|
||||||
^%(PLYR)s\s+(?P<ATYPE>bets|checks|raises|calls|folds|is\sall-In)
|
^%(PLYR)s\s+(?P<ATYPE>bets|checks|raises|calls|folds|is\sall-In)
|
||||||
(?:\s+\[[^.,0-9]?(?P<BET>[.,\d]+)\s+%(CUR)s\])?
|
(?:\s+\[%(CUR_SYM)s(?P<BET>[.,\d]+)\s*%(CUR)s\])?
|
||||||
""" % subst,
|
""" % subst,
|
||||||
re.MULTILINE|re.VERBOSE)
|
re.MULTILINE|re.VERBOSE)
|
||||||
self.re_ShownCards = re.compile(
|
self.re_ShownCards = re.compile(
|
||||||
|
@ -180,7 +188,7 @@ follow : whether to tail -f the input"""
|
||||||
re.MULTILINE)
|
re.MULTILINE)
|
||||||
self.re_CollectPot = re.compile(
|
self.re_CollectPot = re.compile(
|
||||||
r""""^%(PLYR)s \s+ wins \s+
|
r""""^%(PLYR)s \s+ wins \s+
|
||||||
[^.,0-9]?(?P<POT>[.\d]+)\s*%(CUR)s""" % subst,
|
%(CUR_SYM)s(?P<POT>[.\d]+)\s*%(CUR)s""" % subst,
|
||||||
re.MULTILINE|re.VERBOSE)
|
re.MULTILINE|re.VERBOSE)
|
||||||
#self.re_sitsOut = re.compile("^%s sits out" % player_re, re.MULTILINE)
|
#self.re_sitsOut = re.compile("^%s sits out" % player_re, re.MULTILINE)
|
||||||
#self.re_ShownCards = re.compile(
|
#self.re_ShownCards = re.compile(
|
||||||
|
@ -193,11 +201,9 @@ follow : whether to tail -f the input"""
|
||||||
#["ring", "hold", "pl"],
|
#["ring", "hold", "pl"],
|
||||||
#["ring", "hold", "fl"],
|
#["ring", "hold", "fl"],
|
||||||
|
|
||||||
|
|
||||||
["tour", "hold", "nl"],
|
["tour", "hold", "nl"],
|
||||||
#["tour", "hold", "pl"],
|
#["tour", "hold", "pl"],
|
||||||
#["tour", "hold", "fl"],
|
#["tour", "hold", "fl"],
|
||||||
|
|
||||||
]
|
]
|
||||||
|
|
||||||
def _getGameType(self, handText):
|
def _getGameType(self, handText):
|
||||||
|
@ -219,15 +225,10 @@ follow : whether to tail -f the input"""
|
||||||
# inspect the handText and return the gametype dict
|
# inspect the handText and return the gametype dict
|
||||||
# gametype dict is:
|
# gametype dict is:
|
||||||
# {'limitType': xxx, 'base': xxx, 'category': xxx}
|
# {'limitType': xxx, 'base': xxx, 'category': xxx}
|
||||||
#print
|
|
||||||
#print
|
|
||||||
#print '#'*70
|
|
||||||
#print handText
|
|
||||||
#print '#'*70
|
|
||||||
#print
|
|
||||||
#print
|
|
||||||
info = {}
|
|
||||||
|
|
||||||
|
print self.ParsingException().wrapHh( handText )
|
||||||
|
|
||||||
|
info = {}
|
||||||
m = self._getGameType(handText)
|
m = self._getGameType(handText)
|
||||||
if m is None:
|
if m is None:
|
||||||
return None
|
return None
|
||||||
|
@ -241,25 +242,28 @@ follow : whether to tail -f the input"""
|
||||||
games = { # base, category
|
games = { # base, category
|
||||||
"Texas Hold'em" : ('hold','holdem'),
|
"Texas Hold'em" : ('hold','holdem'),
|
||||||
#'Omaha' : ('hold','omahahi'),
|
#'Omaha' : ('hold','omahahi'),
|
||||||
#'Omaha Hi/Lo' : ('hold','omahahilo'),
|
|
||||||
#'Razz' : ('stud','razz'),
|
|
||||||
#'7 Card Stud' : ('stud','studhi'),
|
|
||||||
#'7 Card Stud Hi/Lo' : ('stud','studhilo'),
|
|
||||||
#'Badugi' : ('draw','badugi'),
|
|
||||||
#'Triple Draw 2-7 Lowball' : ('draw','27_3draw'),
|
|
||||||
}
|
}
|
||||||
#currencies = { '$':'USD', '':'T$' }
|
currencies = { '$':'USD', '':'T$' }
|
||||||
# I don't think this is doing what we think. mg will always have all
|
|
||||||
# the expected keys, but the ones that didn't match in the regex will
|
for expectedField in ['LIMIT', 'GAME']:
|
||||||
# have a value of None. It is OK if it throws an exception when it
|
if mg[expectedField] is None:
|
||||||
# runs across an unknown game or limit or whatever.
|
raise self.ParsingException(
|
||||||
if 'LIMIT' in mg:
|
"Cannot fetch field '%s'" % expectedField,
|
||||||
info['limitType'] = limits[mg['LIMIT']]
|
hh = handText)
|
||||||
if 'GAME' in mg:
|
try:
|
||||||
(info['base'], info['category']) = games[mg['GAME']]
|
info['limitType'] = limits[mg['LIMIT']]
|
||||||
|
except:
|
||||||
|
raise self.ParsingException(
|
||||||
|
"Unknown limit '%s'" % mg['LIMIT'],
|
||||||
|
hh = handText)
|
||||||
|
|
||||||
|
try:
|
||||||
|
(info['base'], info['category']) = games[mg['GAME']]
|
||||||
|
except:
|
||||||
|
raise self.ParsingException(
|
||||||
|
"Unknown game type '%s'" % mg['GAME'],
|
||||||
|
hh = handText)
|
||||||
|
|
||||||
if 'CURRENCY' in mg:
|
|
||||||
info['currency'] = mg['CURRENCY']
|
|
||||||
|
|
||||||
if 'TOURNO' in mg:
|
if 'TOURNO' in mg:
|
||||||
info['type'] = 'tour'
|
info['type'] = 'tour'
|
||||||
|
@ -268,9 +272,12 @@ follow : whether to tail -f the input"""
|
||||||
|
|
||||||
if info['type'] == 'ring':
|
if info['type'] == 'ring':
|
||||||
info['sb'], info['bb'] = ringBlinds(mg['RINGLIMIT'])
|
info['sb'], info['bb'] = ringBlinds(mg['RINGLIMIT'])
|
||||||
|
# FIXME: there are only $ and play money availible for cash
|
||||||
|
info['currency'] = currencies(mg['CURRENCY'])
|
||||||
else:
|
else:
|
||||||
info['sb'] = renderTrnyMoney(mg['SB'])
|
info['sb'] = renderTrnyMoney(mg['SB'])
|
||||||
info['bb'] = renderTrnyMoney(mg['BB'])
|
info['bb'] = renderTrnyMoney(mg['BB'])
|
||||||
|
info['currency'] = 'T$'
|
||||||
|
|
||||||
|
|
||||||
# NB: SB, BB must be interpreted as blinds or bets depending on limit type.
|
# NB: SB, BB must be interpreted as blinds or bets depending on limit type.
|
||||||
|
@ -283,10 +290,7 @@ follow : whether to tail -f the input"""
|
||||||
if m:
|
if m:
|
||||||
info.update(m.groupdict())
|
info.update(m.groupdict())
|
||||||
else:
|
else:
|
||||||
print '#'*15, 'START HH', '#'*15
|
raise self.ParsingException("Cannot read Handinfo for current hand", hh=hand.handText)
|
||||||
print hand.handText
|
|
||||||
print '#'*15, ' END HH', '#'*15
|
|
||||||
raise Exception, "Cannot read hand info from hh above"
|
|
||||||
m = self._getGameType(hand.handText)
|
m = self._getGameType(hand.handText)
|
||||||
if m: info.update(m.groupdict())
|
if m: info.update(m.groupdict())
|
||||||
m = self.re_Hid.search(hand.handText)
|
m = self.re_Hid.search(hand.handText)
|
||||||
|
@ -318,20 +322,16 @@ follow : whether to tail -f the input"""
|
||||||
hand.tablename = info[key]
|
hand.tablename = info[key]
|
||||||
if key == 'BUTTON':
|
if key == 'BUTTON':
|
||||||
hand.buttonpos = info[key]
|
hand.buttonpos = info[key]
|
||||||
#if key == 'MAX':
|
|
||||||
#hand.maxseats = int(info[key])
|
|
||||||
|
|
||||||
#if key == 'MIXED':
|
|
||||||
#if info[key] == None: hand.mixed = None
|
|
||||||
#else: hand.mixed = self.mixes[info[key]]
|
|
||||||
|
|
||||||
if key == 'TOURNO':
|
if key == 'TOURNO':
|
||||||
hand.tourNo = info[key]
|
hand.tourNo = info[key]
|
||||||
if key == 'BUYIN':
|
if key == 'BUYIN':
|
||||||
hand.buyin = info[key]
|
#FIXME: it's dirty hack T_T
|
||||||
|
cur = info[key][0] if info[key][0] not in '0123456789' else ''
|
||||||
|
hand.buyin = info[key] + '+%s0' % cur
|
||||||
if key == 'LEVEL':
|
if key == 'LEVEL':
|
||||||
hand.level = info[key]
|
hand.level = info[key]
|
||||||
if key == 'PLAY' and info['PLAY'] != 'Real':
|
if key == 'PLAY' and info['PLAY'] != 'Real':
|
||||||
|
# TODO: play money wasn't tested
|
||||||
# hand.currency = 'play' # overrides previously set value
|
# hand.currency = 'play' # overrides previously set value
|
||||||
hand.gametype['currency'] = 'play'
|
hand.gametype['currency'] = 'play'
|
||||||
|
|
||||||
|
@ -364,9 +364,8 @@ follow : whether to tail -f the input"""
|
||||||
, hand.handText,re.DOTALL)
|
, hand.handText,re.DOTALL)
|
||||||
hand.addStreets(m)
|
hand.addStreets(m)
|
||||||
|
|
||||||
def readCommunityCards(self, hand, street): # street has been matched by markStreets, so exists in this hand
|
def readCommunityCards(self, hand, street):
|
||||||
if street in ('FLOP','TURN','RIVER'): # a list of streets which get dealt community cards (i.e. all but PREFLOP)
|
if street in ('FLOP','TURN','RIVER'):
|
||||||
#print "DEBUG readCommunityCards:", street, hand.streets.group(street)
|
|
||||||
m = self.re_Board.search(hand.streets[street])
|
m = self.re_Board.search(hand.streets[street])
|
||||||
hand.setCommunityCards(street, renderCards(m.group('CARDS')))
|
hand.setCommunityCards(street, renderCards(m.group('CARDS')))
|
||||||
|
|
||||||
|
@ -399,10 +398,10 @@ follow : whether to tail -f the input"""
|
||||||
# party doesn't track blinds for tournaments
|
# party doesn't track blinds for tournaments
|
||||||
# so there're some cra^Wcaclulations
|
# so there're some cra^Wcaclulations
|
||||||
if hand.buttonpos == 0:
|
if hand.buttonpos == 0:
|
||||||
self.readButton(self, hand)
|
self.readButton(hand)
|
||||||
# NOTE: code below depends on Hand's implementation
|
# NOTE: code below depends on Hand's implementation
|
||||||
# playersMap - dict {seat: (pname,stack)}
|
# playersMap - dict {seat: (pname,stack)}
|
||||||
playersMap = dict([(f[0], f[1:2]) for f in hand.players])
|
playersMap = dict([(f[0], f[1:3]) for f in hand.players])
|
||||||
maxSeat = max(playersMap)
|
maxSeat = max(playersMap)
|
||||||
|
|
||||||
def findFirstNonEmptySeat(startSeat):
|
def findFirstNonEmptySeat(startSeat):
|
||||||
|
@ -416,7 +415,7 @@ follow : whether to tail -f the input"""
|
||||||
if noSmallBlind:
|
if noSmallBlind:
|
||||||
hand.addBlind(None, None, None)
|
hand.addBlind(None, None, None)
|
||||||
else:
|
else:
|
||||||
smallBlindSeat = findFirstNonEmptySeat(hand.buttonpos + 1)
|
smallBlindSeat = findFirstNonEmptySeat(int(hand.buttonpos) + 1)
|
||||||
blind = smartMin(hand.sb, playersMap[smallBlindSeat][1])
|
blind = smartMin(hand.sb, playersMap[smallBlindSeat][1])
|
||||||
hand.addBlind(playersMap[smallBlindSeat][0], 'small blind', blind)
|
hand.addBlind(playersMap[smallBlindSeat][0], 'small blind', blind)
|
||||||
|
|
||||||
|
@ -446,6 +445,8 @@ follow : whether to tail -f the input"""
|
||||||
for action in m:
|
for action in m:
|
||||||
acts = action.groupdict()
|
acts = action.groupdict()
|
||||||
if action.group('ATYPE') in ('raises','is all-In'):
|
if action.group('ATYPE') in ('raises','is all-In'):
|
||||||
|
#print action.groupdict()
|
||||||
|
#sys.exit(1)
|
||||||
hand.addRaiseBy( street, action.group('PNAME'), action.group('BET') )
|
hand.addRaiseBy( street, action.group('PNAME'), action.group('BET') )
|
||||||
elif action.group('ATYPE') == 'calls':
|
elif action.group('ATYPE') == 'calls':
|
||||||
hand.addCall( street, action.group('PNAME'), action.group('BET') )
|
hand.addCall( street, action.group('PNAME'), action.group('BET') )
|
||||||
|
|
Loading…
Reference in New Issue
Block a user