2010-07-04 03:05:16 +02:00
#!/usr/bin/env python2
2009-03-10 00:03:17 +01:00
# -*- coding: utf-8 -*-
#
2010-07-04 03:05:16 +02:00
# Copyright 2008-2010, Carl Gherardi
2009-03-06 02:24:38 +01:00
#
# 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
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
########################################################################
2009-07-13 06:37:51 +02:00
# TODO: straighten out discards for draw games
2009-08-09 16:19:43 +02:00
2009-03-06 02:24:38 +01:00
import sys
from HandHistoryConverter import *
# PokerStars HH Format
class PokerStars ( HandHistoryConverter ) :
2009-03-15 06:32:52 +01:00
2009-08-09 16:19:43 +02:00
# Class Variables
sitename = " PokerStars "
filetype = " text "
2009-11-22 05:34:11 +01:00
codepage = ( " utf8 " , " cp1252 " )
2009-08-09 16:19:43 +02:00
siteId = 2 # Needs to match id entry in Sites database
2009-07-13 06:37:51 +02:00
2009-08-01 17:51:37 +02:00
mixes = { ' HORSE ' : ' horse ' , ' 8-Game ' : ' 8game ' , ' HOSE ' : ' hose ' } # Legal mixed games
2009-11-22 05:34:11 +01:00
sym = { ' USD ' : " \ $ " , ' CAD ' : " \ $ " , ' T$ ' : " " , " EUR " : " \xe2 \x82 \xac " , " GBP " : " \xa3 " } # ADD Euro, Sterling, etc HERE
2009-07-31 00:06:13 +02:00
substitutions = {
2009-08-07 01:27:52 +02:00
' LEGAL_ISO ' : " USD|EUR|GBP|CAD|FPP " , # legal ISO currency codes
2009-11-22 05:34:11 +01:00
' LS ' : " \ $| \xe2 \x82 \xac | " # legal currency symbols - Euro(cp1252, utf-8)
2009-07-31 00:06:13 +02:00
}
2009-03-06 02:24:38 +01:00
# Static regexes
2009-11-22 05:34:11 +01:00
re_GameInfo = re . compile ( u """
2009-07-31 00:06:13 +02:00
PokerStars \sGame \s \#(?P<HID>[0-9]+):\s+
( Tournament \s \# # open paren of tournament info
( ? P < TOURNO > \d + ) , \s
2010-01-22 23:42:44 +01:00
# here's how I plan to use LS
( ? P < BUYIN > ( [ % ( LS ) s \+ \d \. ] + \s ? ( ? P < TOUR_ISO > % ( LEGAL_ISO ) s ) ? ) | Freeroll ) \s + ) ?
# close paren of tournament info
2009-07-31 00:06:13 +02:00
( ? P < MIXED > HORSE | 8 \- Game | HOSE ) ? \s ? \( ?
2010-03-30 07:04:39 +02:00
( ? P < GAME > Hold \' em|Razz|RAZZ|7 \ sCard \ sStud|7 \ sCard \ sStud \ sHi/Lo|Omaha|Omaha \ sHi/Lo|Badugi|Triple \ sDraw \ s2 \ -7 \ sLowball|5 \ sCard \ sDraw) \ s
( ? P < LIMIT > No \sLimit | Limit | LIMIT | Pot \sLimit ) \) ? , ? \s
2009-07-31 00:06:13 +02:00
( - \sLevel \s ( ? P < LEVEL > [ IVXLC ] + ) \s ) ?
\( ? # open paren of the stakes
( ? P < CURRENCY > % ( LS ) s | ) ?
2009-08-03 05:50:44 +02:00
( ? P < SB > [ .0 - 9 ] + ) / ( % ( LS ) s ) ?
2009-07-31 00:06:13 +02:00
( ? P < BB > [ .0 - 9 ] + )
\s ? ( ? P < ISO > % ( LEGAL_ISO ) s ) ?
\) \s - \s # close paren of the stakes
( ? P < DATETIME > . * $ ) """ % s ubstitutions,
re . MULTILINE | re . VERBOSE )
2009-08-01 17:51:37 +02:00
2009-11-22 05:34:11 +01:00
re_PlayerInfo = re . compile ( u """
2009-08-01 17:51:37 +02:00
^ Seat \s ( ? P < SEAT > [ 0 - 9 ] + ) : \s
( ? P < PNAME > . * ) \s
2009-08-03 05:50:44 +02:00
\( ( % ( LS ) s ) ? ( ? P < CASH > [ .0 - 9 ] + ) \sin \schips \) """ % s ubstitutions,
2009-08-01 17:51:37 +02:00
re . MULTILINE | re . VERBOSE )
re_HandInfo = re . compile ( """
2010-02-03 14:52:44 +01:00
^ Table \s \' (?P<TABLE>[- \ \ #a-zA-Z \ d]+) \' \ s
2009-08-01 17:51:37 +02:00
( ( ? P < MAX > \d + ) - max \s ) ?
( ? P < PLAY > \( Play \sMoney \) \s ) ?
( Seat \s \#(?P<BUTTON>\d+)\sis\sthe\sbutton)?""",
re . MULTILINE | re . VERBOSE )
2009-03-31 08:44:19 +02:00
re_SplitHands = re . compile ( ' \n \n + ' )
re_TailSplitHands = re . compile ( ' ( \n \n \n +) ' )
2009-03-06 02:24:38 +01:00
re_Button = re . compile ( ' Seat #(?P<BUTTON> \ d+) is the button ' , re . MULTILINE )
re_Board = re . compile ( r " \ [(?P<CARDS>.+) \ ] " )
# self.re_setHandInfoRegex('.*#(?P<HID>[0-9]+): Table (?P<TABLE>[ a-zA-Z]+) - \$?(?P<SB>[.0-9]+)/\$?(?P<BB>[.0-9]+) - (?P<GAMETYPE>.*) - (?P<HR>[0-9]+):(?P<MIN>[0-9]+) ET - (?P<YEAR>[0-9]+)/(?P<MON>[0-9]+)/(?P<DAY>[0-9]+)Table (?P<TABLE>[ a-zA-Z]+)\nSeat (?P<BUTTON>[0-9]+)')
2009-07-13 22:21:20 +02:00
2010-01-17 19:54:40 +01:00
re_DateTime = re . compile ( """ (?P<Y>[0-9] {4} ) \ /(?P<M>[0-9] {2} ) \ /(?P<D>[0-9] {2} )[ \ - ]+(?P<H>[0-9]+):(?P<MIN>[0-9]+):(?P<S>[0-9]+) """ , re . MULTILINE )
2010-03-16 23:32:31 +01:00
# revised re including timezone (not currently used):
#re_DateTime = re.compile("""(?P<Y>[0-9]{4})\/(?P<M>[0-9]{2})\/(?P<D>[0-9]{2})[\- ]+(?P<H>[0-9]+):(?P<MIN>[0-9]+):(?P<S>[0-9]+) \(?(?P<TZ>[A-Z0-9]+)""", re.MULTILINE)
2009-07-13 22:21:20 +02:00
2009-03-10 00:03:17 +01:00
def compilePlayerRegexs ( self , hand ) :
2009-03-10 17:17:54 +01:00
players = set ( [ player [ 1 ] for player in hand . players ] )
2009-03-06 02:24:38 +01:00
if not players < = self . compiledPlayers : # x <= y means 'x is subset of y'
# we need to recompile the player regexs.
2009-07-13 06:37:51 +02:00
# TODO: should probably rename re_HeroCards and corresponding method,
# since they are used to find all cards on lines starting with "Dealt to:"
# They still identify the hero.
2009-03-06 02:24:38 +01:00
self . compiledPlayers = players
player_re = " (?P<PNAME> " + " | " . join ( map ( re . escape , players ) ) + " ) "
2009-08-01 17:51:37 +02:00
subst = { ' PLYR ' : player_re , ' CUR ' : self . sym [ hand . gametype [ ' currency ' ] ] }
2009-11-10 01:24:46 +01:00
log . debug ( " player_re: " + player_re )
2009-08-01 17:51:37 +02:00
self . re_PostSB = re . compile ( r " ^ %(PLYR)s : posts small blind %(CUR)s (?P<SB>[.0-9]+) " % subst , re . MULTILINE )
self . re_PostBB = re . compile ( r " ^ %(PLYR)s : posts big blind %(CUR)s (?P<BB>[.0-9]+) " % subst , re . MULTILINE )
self . re_Antes = re . compile ( r " ^ %(PLYR)s : posts the ante %(CUR)s (?P<ANTE>[.0-9]+) " % subst , re . MULTILINE )
self . re_BringIn = re . compile ( r " ^ %(PLYR)s : brings[- ]in( low|) for %(CUR)s (?P<BRINGIN>[.0-9]+) " % subst , re . MULTILINE )
2010-01-17 19:54:40 +01:00
self . re_PostBoth = re . compile ( r " ^ %(PLYR)s : posts small \ & big blinds %(CUR)s (?P<SBBB>[.0-9]+) " % subst , re . MULTILINE )
2009-08-01 17:51:37 +02:00
self . re_HeroCards = re . compile ( r " ^Dealt to %(PLYR)s (?: \ [(?P<OLDCARDS>.+?) \ ])?( \ [(?P<NEWCARDS>.+?) \ ]) " % subst , re . MULTILINE )
self . re_Action = re . compile ( r """
2010-02-15 00:27:34 +01:00
^ % ( PLYR ) s : ( ? P < ATYPE > \sbets | \schecks | \sraises | \scalls | \sfolds | \sdiscards | \sstands \spat )
2009-11-29 07:40:32 +01:00
( \s ( % ( CUR ) s ) ? ( ? P < BET > [ . \d ] + ) ) ? ( \sto \s % ( CUR ) s ( ? P < BETTO > [ . \d ] + ) ) ? # the number discarded goes in <BET>
2010-02-15 00:27:34 +01:00
\s * ( and \sis \sall . in ) ?
( \scards ? ( \s \[ ( ? P < DISCARDED > . + ? ) \] ) ? ) ? \s * $ """
2009-08-01 17:51:37 +02:00
% subst , re . MULTILINE | re . VERBOSE )
2009-03-06 02:24:38 +01:00
self . re_ShowdownAction = re . compile ( r " ^ %s : shows \ [(?P<CARDS>.*) \ ] " % player_re , re . MULTILINE )
2010-01-17 19:54:40 +01:00
self . re_CollectPot = re . compile ( r " Seat (?P<SEAT>[0-9]+): %(PLYR)s ( \ (button \ ) | \ (small blind \ ) | \ (big blind \ ) | \ (button \ ) \ (small blind \ ) )?(collected|showed \ [.* \ ] and won) \ ( %(CUR)s (?P<POT>[. \ d]+) \ )(, mucked| with.*|) " % subst , re . MULTILINE )
2009-03-06 02:24:38 +01:00
self . re_sitsOut = re . compile ( " ^ %s sits out " % player_re , re . MULTILINE )
2009-07-04 02:41:08 +02:00
self . re_ShownCards = re . compile ( " ^Seat (?P<SEAT>[0-9]+): %s ( \ (.* \ ) )?(?P<SHOWED>showed|mucked) \ [(?P<CARDS>.*) \ ].* " % player_re , re . MULTILINE )
2009-03-06 02:24:38 +01:00
def readSupportedGames ( self ) :
2009-03-14 11:48:34 +01:00
return [ [ " ring " , " hold " , " nl " ] ,
[ " ring " , " hold " , " pl " ] ,
[ " ring " , " hold " , " fl " ] ,
2009-07-07 19:48:43 +02:00
2009-03-14 11:48:34 +01:00
[ " ring " , " stud " , " fl " ] ,
2009-07-11 19:44:32 +02:00
[ " ring " , " draw " , " fl " ] ,
2009-07-07 19:48:43 +02:00
[ " tour " , " hold " , " nl " ] ,
[ " tour " , " hold " , " pl " ] ,
[ " tour " , " hold " , " fl " ] ,
[ " tour " , " stud " , " fl " ] ,
2009-03-14 11:48:34 +01:00
]
2009-03-06 02:24:38 +01:00
def determineGameType ( self , handText ) :
2009-07-11 19:44:32 +02:00
# inspect the handText and return the gametype dict
# gametype dict is:
# {'limitType': xxx, 'base': xxx, 'category': xxx}
2009-03-06 02:24:38 +01:00
2009-07-07 19:48:43 +02:00
info = { }
2009-03-06 02:24:38 +01:00
m = self . re_GameInfo . search ( handText )
2009-03-15 06:32:52 +01:00
if not m :
2010-04-22 18:33:24 +02:00
tmp = handText [ 0 : 100 ]
log . error ( " determineGameType: Unable to recognise gametype from: ' %s ' " % tmp )
log . error ( " determineGameType: Raising FpdbParseError " )
2010-06-04 09:59:47 +02:00
raise FpdbParseError ( " Unable to recognise gametype from: ' %s ' " % tmp )
2009-03-10 00:03:17 +01:00
mg = m . groupdict ( )
2009-07-11 19:44:32 +02:00
# translations from captured groups to fpdb info strings
2010-02-18 22:12:01 +01:00
Lim_Blinds = { ' 0.04 ' : ( ' 0.01 ' , ' 0.02 ' ) , ' 0.10 ' : ( ' 0.02 ' , ' 0.05 ' ) , ' 0.20 ' : ( ' 0.05 ' , ' 0.10 ' ) ,
2010-06-25 10:58:14 +02:00
' 0.40 ' : ( ' 0.10 ' , ' 0.20 ' ) , ' 0.50 ' : ( ' 0.10 ' , ' 0.25 ' ) , ' 1.00 ' : ( ' 0.25 ' , ' 0.50 ' ) ,
' 2.00 ' : ( ' 0.50 ' , ' 1.00 ' ) , ' 2 ' : ( ' 0.50 ' , ' 1.00 ' ) , ' 4 ' : ( ' 1.00 ' , ' 2.00 ' ) ,
' 4.00 ' : ( ' 1.00 ' , ' 2.00 ' ) , ' 6 ' : ( ' 1.00 ' , ' 3.00 ' ) , ' 6.00 ' : ( ' 1.00 ' , ' 3.00 ' ) ,
' 10.00 ' : ( ' 2.00 ' , ' 5.00 ' ) , ' 20.00 ' : ( ' 5.00 ' , ' 10.00 ' ) , ' 30.00 ' : ( ' 10.00 ' , ' 15.00 ' ) ,
' 60.00 ' : ( ' 15.00 ' , ' 30.00 ' ) , ' 100.00 ' : ( ' 25.00 ' , ' 50.00 ' ) , ' 200.00 ' : ( ' 50.00 ' , ' 100.00 ' ) ,
' 400.00 ' : ( ' 100.00 ' , ' 200.00 ' ) , ' 1000.00 ' : ( ' 250.00 ' , ' 500.00 ' ) }
2010-02-18 22:12:01 +01:00
2010-03-30 07:04:39 +02:00
limits = { ' No Limit ' : ' nl ' , ' Pot Limit ' : ' pl ' , ' Limit ' : ' fl ' , ' LIMIT ' : ' fl ' }
2009-07-11 19:44:32 +02:00
games = { # base, category
" Hold ' em " : ( ' hold ' , ' holdem ' ) ,
' Omaha ' : ( ' hold ' , ' omahahi ' ) ,
' Omaha Hi/Lo ' : ( ' hold ' , ' omahahilo ' ) ,
' Razz ' : ( ' stud ' , ' razz ' ) ,
2010-04-22 18:33:24 +02:00
' RAZZ ' : ( ' stud ' , ' razz ' ) ,
2009-07-11 19:44:32 +02:00
' 7 Card Stud ' : ( ' stud ' , ' studhi ' ) ,
' 7 Card Stud Hi/Lo ' : ( ' stud ' , ' studhilo ' ) ,
' Badugi ' : ( ' draw ' , ' badugi ' ) ,
' Triple Draw 2-7 Lowball ' : ( ' draw ' , ' 27_3draw ' ) ,
2009-11-29 09:54:15 +01:00
' 5 Card Draw ' : ( ' draw ' , ' fivedraw ' )
2009-03-10 00:03:17 +01:00
}
currencies = { u ' € ' : ' EUR ' , ' $ ' : ' USD ' , ' ' : ' T$ ' }
2009-07-07 19:48:43 +02:00
# 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
# have a value of None. It is OK if it throws an exception when it
# runs across an unknown game or limit or whatever.
2009-03-10 00:03:17 +01:00
if ' LIMIT ' in mg :
info [ ' limitType ' ] = limits [ mg [ ' LIMIT ' ] ]
if ' GAME ' in mg :
( info [ ' base ' ] , info [ ' category ' ] ) = games [ mg [ ' GAME ' ] ]
if ' SB ' in mg :
info [ ' sb ' ] = mg [ ' SB ' ]
if ' BB ' in mg :
info [ ' bb ' ] = mg [ ' BB ' ]
if ' CURRENCY ' in mg :
info [ ' currency ' ] = currencies [ mg [ ' CURRENCY ' ] ]
2009-07-11 19:44:32 +02:00
2009-11-03 20:30:52 +01:00
if ' TOURNO ' in mg and mg [ ' TOURNO ' ] is None :
2009-07-07 19:48:43 +02:00
info [ ' type ' ] = ' ring '
else :
info [ ' type ' ] = ' tour '
2009-07-11 19:44:32 +02:00
2010-03-04 17:50:03 +01:00
if info [ ' limitType ' ] == ' fl ' and info [ ' bb ' ] is not None and info [ ' type ' ] == ' ring ' and info [ ' base ' ] != ' stud ' :
2010-04-22 18:33:24 +02:00
try :
info [ ' sb ' ] = Lim_Blinds [ mg [ ' BB ' ] ] [ 0 ]
info [ ' bb ' ] = Lim_Blinds [ mg [ ' BB ' ] ] [ 1 ]
except KeyError :
log . error ( " determineGameType: Lim_Blinds has no lookup for ' %s ' " % mg [ ' BB ' ] )
log . error ( " determineGameType: Raising FpdbParseError " )
2010-06-04 09:59:47 +02:00
raise FpdbParseError ( " Lim_Blinds has no lookup for ' %s ' " % mg [ ' BB ' ] )
2010-02-18 22:12:01 +01:00
2009-03-10 00:03:17 +01:00
return info
2009-03-06 02:24:38 +01:00
def readHandInfo ( self , hand ) :
info = { }
m = self . re_HandInfo . search ( hand . handText , re . DOTALL )
2009-03-14 16:30:38 +01:00
if m :
info . update ( m . groupdict ( ) )
2009-07-11 19:44:32 +02:00
# hand.maxseats = int(m2.group(1))
else :
pass # throw an exception here, eh?
2009-03-06 02:24:38 +01:00
m = self . re_GameInfo . search ( hand . handText )
2009-11-03 20:30:52 +01:00
if m :
info . update ( m . groupdict ( ) )
2009-07-11 19:44:32 +02:00
# m = self.re_Button.search(hand.handText)
# if m: info.update(m.groupdict())
2009-03-06 02:24:38 +01:00
# TODO : I rather like the idea of just having this dict as hand.info
2009-11-10 01:24:46 +01:00
log . debug ( " readHandInfo: %s " % info )
2009-03-06 02:24:38 +01:00
for key in info :
if key == ' DATETIME ' :
2010-03-16 23:32:31 +01:00
#2008/11/12 10:00:48 CET [2008/11/12 4:00:48 ET] # (both dates are parsed so ET date overrides the other)
2009-03-25 10:35:19 +01:00
#2008/08/17 - 01:14:43 (ET)
#2008/09/07 06:23:14 ET
2010-01-17 19:54:40 +01:00
m1 = self . re_DateTime . finditer ( info [ key ] )
# m2 = re.search("(?P<Y>[0-9]{4})\/(?P<M>[0-9]{2})\/(?P<D>[0-9]{2})[\- ]+(?P<H>[0-9]+):(?P<MIN>[0-9]+):(?P<S>[0-9]+)", info[key])
2010-03-16 23:32:31 +01:00
datetimestr = " 2000/01/01 00:00:00 " # default used if time not found (stops import crashing, but handstart will be wrong)
2010-01-17 19:54:40 +01:00
for a in m1 :
datetimestr = " %s / %s / %s %s : %s : %s " % ( a . group ( ' Y ' ) , a . group ( ' M ' ) , a . group ( ' D ' ) , a . group ( ' H ' ) , a . group ( ' MIN ' ) , a . group ( ' S ' ) )
2010-03-16 23:32:31 +01:00
#tz = a.group('TZ') # just assume ET??
#print " tz = ", tz, " datetime =", datetimestr
hand . starttime = datetime . datetime . strptime ( datetimestr , " % Y/ % m/ %d % H: % M: % S " ) # also timezone at end, e.g. " ET"
# approximate rules for ET daylight savings time:
if ( hand . starttime . month == 12 # all of Dec
or ( hand . starttime . month == 11 and hand . starttime . day > 4 ) # and most of November
or hand . starttime . month < 3 # and all of Jan/Feb
or ( hand . starttime . month == 3 and hand . starttime . day < 11 ) ) : # and 1st 10 days of March
offset = datetime . timedelta ( hours = 5 ) # are EST: assume 5 hour offset (ET without daylight saving)
else :
offset = datetime . timedelta ( hours = 4 ) # rest is EDT: assume 4 hour offset (ET with daylight saving)
# adjust time into UTC:
hand . starttime = hand . starttime + offset
#print " tz = %s start = %s" % (tz, str(hand.starttime))
2009-03-06 02:24:38 +01:00
if key == ' HID ' :
hand . handid = info [ key ]
2009-11-05 17:27:33 +01:00
if key == ' TOURNO ' :
hand . tourNo = info [ key ]
if key == ' BUYIN ' :
2010-01-22 23:42:44 +01:00
if info [ key ] == ' Freeroll ' :
hand . buyin = ' $0+$0 '
else :
#FIXME: The key looks like: '€0.82+€0.18 EUR'
# This should be parsed properly and used
hand . buyin = info [ key ]
2009-11-05 17:27:33 +01:00
if key == ' LEVEL ' :
hand . level = info [ key ]
2009-03-06 02:24:38 +01:00
if key == ' TABLE ' :
2009-11-05 17:27:33 +01:00
if hand . tourNo != None :
hand . tablename = re . split ( " " , info [ key ] ) [ 1 ]
else :
hand . tablename = info [ key ]
2009-03-06 02:24:38 +01:00
if key == ' BUTTON ' :
hand . buttonpos = info [ key ]
2009-07-11 19:44:32 +02:00
if key == ' MAX ' :
2009-07-12 19:14:50 +02:00
hand . maxseats = int ( info [ key ] )
2009-07-11 19:44:32 +02:00
if key == ' MIXED ' :
2009-11-03 20:30:52 +01:00
hand . mixed = self . mixes [ info [ key ] ] if info [ key ] is not None else None
if key == ' PLAY ' and info [ ' PLAY ' ] is not None :
2009-07-13 22:21:20 +02:00
# hand.currency = 'play' # overrides previously set value
hand . gametype [ ' currency ' ] = ' play '
2009-07-11 19:44:32 +02:00
2009-03-06 02:24:38 +01:00
def readButton ( self , hand ) :
m = self . re_Button . search ( hand . handText )
if m :
hand . buttonpos = int ( m . group ( ' BUTTON ' ) )
else :
2009-11-10 01:24:46 +01:00
log . info ( ' readButton: not found ' )
2009-03-06 02:24:38 +01:00
def readPlayerStacks ( self , hand ) :
2009-11-10 01:24:46 +01:00
log . debug ( " readPlayerStacks " )
2009-03-06 02:24:38 +01:00
m = self . re_PlayerInfo . finditer ( hand . handText )
for a in m :
hand . addPlayer ( int ( a . group ( ' SEAT ' ) ) , a . group ( ' PNAME ' ) , a . group ( ' CASH ' ) )
def markStreets ( self , hand ) :
# PREFLOP = ** Dealing down cards **
# This re fails if, say, river is missing; then we don't get the ** that starts the river.
2009-03-11 15:42:49 +01:00
if hand . gametype [ ' base ' ] in ( " hold " ) :
2009-03-06 02:24:38 +01:00
m = re . search ( r " \ * \ * \ * HOLE CARDS \ * \ * \ *(?P<PREFLOP>.+(?= \ * \ * \ * FLOP \ * \ * \ *)|.+) "
r " ( \ * \ * \ * FLOP \ * \ * \ *(?P<FLOP> \ [ \ S \ S \ S \ S \ S \ S \ ].+(?= \ * \ * \ * TURN \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * TURN \ * \ * \ * \ [ \ S \ S \ S \ S \ S \ S] (?P<TURN> \ [ \ S \ S \ ].+(?= \ * \ * \ * RIVER \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * RIVER \ * \ * \ * \ [ \ S \ S \ S \ S \ S \ S \ S \ S] (?P<RIVER> \ [ \ S \ S \ ].+))? " , hand . handText , re . DOTALL )
2009-03-11 15:42:49 +01:00
elif hand . gametype [ ' base ' ] in ( " stud " ) :
2009-03-06 02:24:38 +01:00
m = re . search ( r " (?P<ANTES>.+(?= \ * \ * \ * 3rd STREET \ * \ * \ *)|.+) "
r " ( \ * \ * \ * 3rd STREET \ * \ * \ *(?P<THIRD>.+(?= \ * \ * \ * 4th STREET \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * 4th STREET \ * \ * \ *(?P<FOURTH>.+(?= \ * \ * \ * 5th STREET \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * 5th STREET \ * \ * \ *(?P<FIFTH>.+(?= \ * \ * \ * 6th STREET \ * \ * \ *)|.+))? "
2009-03-11 19:40:17 +01:00
r " ( \ * \ * \ * 6th STREET \ * \ * \ *(?P<SIXTH>.+(?= \ * \ * \ * RIVER \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * RIVER \ * \ * \ *(?P<SEVENTH>.+))? " , hand . handText , re . DOTALL )
2009-03-14 11:48:34 +01:00
elif hand . gametype [ ' base ' ] in ( " draw " ) :
m = re . search ( r " (?P<PREDEAL>.+(?= \ * \ * \ * DEALING HANDS \ * \ * \ *)|.+) "
r " ( \ * \ * \ * DEALING HANDS \ * \ * \ *(?P<DEAL>.+(?= \ * \ * \ * FIRST DRAW \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * FIRST DRAW \ * \ * \ *(?P<DRAWONE>.+(?= \ * \ * \ * SECOND DRAW \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * SECOND DRAW \ * \ * \ *(?P<DRAWTWO>.+(?= \ * \ * \ * THIRD DRAW \ * \ * \ *)|.+))? "
r " ( \ * \ * \ * THIRD DRAW \ * \ * \ *(?P<DRAWTHREE>.+))? " , hand . handText , re . DOTALL )
2009-03-06 02:24:38 +01:00
hand . addStreets ( m )
def readCommunityCards ( self , hand , street ) : # street has been matched by markStreets, so exists in this hand
if street in ( ' FLOP ' , ' TURN ' , ' RIVER ' ) : # a list of streets which get dealt community cards (i.e. all but PREFLOP)
#print "DEBUG readCommunityCards:", street, hand.streets.group(street)
m = self . re_Board . search ( hand . streets [ street ] )
hand . setCommunityCards ( street , m . group ( ' CARDS ' ) . split ( ' ' ) )
def readAntes ( self , hand ) :
2009-11-10 01:24:46 +01:00
log . debug ( " reading antes " )
2009-03-06 02:24:38 +01:00
m = self . re_Antes . finditer ( hand . handText )
for player in m :
2009-03-11 19:40:17 +01:00
#~ logging.debug("hand.addAnte(%s,%s)" %(player.group('PNAME'), player.group('ANTE')))
2009-03-06 02:24:38 +01:00
hand . addAnte ( player . group ( ' PNAME ' ) , player . group ( ' ANTE ' ) )
def readBringIn ( self , hand ) :
m = self . re_BringIn . search ( hand . handText , re . DOTALL )
if m :
2009-03-11 19:40:17 +01:00
#~ logging.debug("readBringIn: %s for %s" %(m.group('PNAME'), m.group('BRINGIN')))
2009-03-06 02:24:38 +01:00
hand . addBringIn ( m . group ( ' PNAME ' ) , m . group ( ' BRINGIN ' ) )
def readBlinds ( self , hand ) :
2010-02-18 22:12:01 +01:00
liveBlind = True
for a in self . re_PostSB . finditer ( hand . handText ) :
if liveBlind :
hand . addBlind ( a . group ( ' PNAME ' ) , ' small blind ' , a . group ( ' SB ' ) )
liveBlind = False
else :
# Post dead blinds as ante
hand . addBlind ( a . group ( ' PNAME ' ) , ' secondsb ' , a . group ( ' SB ' ) )
2009-03-06 02:24:38 +01:00
for a in self . re_PostBB . finditer ( hand . handText ) :
hand . addBlind ( a . group ( ' PNAME ' ) , ' big blind ' , a . group ( ' BB ' ) )
for a in self . re_PostBoth . finditer ( hand . handText ) :
2009-06-19 08:21:35 +02:00
hand . addBlind ( a . group ( ' PNAME ' ) , ' both ' , a . group ( ' SBBB ' ) )
2009-03-06 02:24:38 +01:00
def readHeroCards ( self , hand ) :
2009-07-13 06:37:51 +02:00
# streets PREFLOP, PREDRAW, and THIRD are special cases beacause
# we need to grab hero's cards
for street in ( ' PREFLOP ' , ' DEAL ' ) :
2009-07-12 22:01:02 +02:00
if street in hand . streets . keys ( ) :
2009-07-13 06:37:51 +02:00
m = self . re_HeroCards . finditer ( hand . streets [ street ] )
for found in m :
# if m == None:
# hand.involved = False
# else:
hand . hero = found . group ( ' PNAME ' )
newcards = found . group ( ' NEWCARDS ' ) . split ( ' ' )
2009-07-12 22:01:02 +02:00
hand . addHoleCards ( street , hand . hero , closed = newcards , shown = False , mucked = False , dealt = True )
2009-07-13 06:37:51 +02:00
for street , text in hand . streets . iteritems ( ) :
2009-07-15 01:26:53 +02:00
if not text or street in ( ' PREFLOP ' , ' DEAL ' ) : continue # already done these
2009-07-13 06:37:51 +02:00
m = self . re_HeroCards . finditer ( hand . streets [ street ] )
for found in m :
player = found . group ( ' PNAME ' )
2009-11-03 20:30:52 +01:00
if found . group ( ' NEWCARDS ' ) is None :
2009-07-13 06:37:51 +02:00
newcards = [ ]
2009-03-14 11:48:34 +01:00
else :
2009-07-13 06:37:51 +02:00
newcards = found . group ( ' NEWCARDS ' ) . split ( ' ' )
2009-11-03 20:30:52 +01:00
if found . group ( ' OLDCARDS ' ) is None :
2009-07-13 06:37:51 +02:00
oldcards = [ ]
2009-03-14 11:48:34 +01:00
else :
2009-07-13 06:37:51 +02:00
oldcards = found . group ( ' OLDCARDS ' ) . split ( ' ' )
2009-03-14 11:48:34 +01:00
2009-07-13 06:37:51 +02:00
if street == ' THIRD ' and len ( newcards ) == 3 : # hero in stud game
hand . hero = player
hand . dealt . add ( player ) # need this for stud??
hand . addHoleCards ( street , player , closed = newcards [ 0 : 2 ] , open = [ newcards [ 2 ] ] , shown = False , mucked = False , dealt = False )
else :
hand . addHoleCards ( street , player , open = newcards , closed = oldcards , shown = False , mucked = False , dealt = False )
2009-03-14 11:48:34 +01:00
2009-03-06 02:24:38 +01:00
def readAction ( self , hand , street ) :
m = self . re_Action . finditer ( hand . streets [ street ] )
for action in m :
2009-07-11 19:44:32 +02:00
acts = action . groupdict ( )
2010-02-14 16:54:19 +01:00
#print "DEBUG: acts: %s" %acts
2009-03-06 02:24:38 +01:00
if action . group ( ' ATYPE ' ) == ' raises ' :
hand . addRaiseBy ( street , action . group ( ' PNAME ' ) , action . group ( ' BET ' ) )
elif action . group ( ' ATYPE ' ) == ' calls ' :
hand . addCall ( street , action . group ( ' PNAME ' ) , action . group ( ' BET ' ) )
elif action . group ( ' ATYPE ' ) == ' bets ' :
hand . addBet ( street , action . group ( ' PNAME ' ) , action . group ( ' BET ' ) )
elif action . group ( ' ATYPE ' ) == ' folds ' :
hand . addFold ( street , action . group ( ' PNAME ' ) )
elif action . group ( ' ATYPE ' ) == ' checks ' :
hand . addCheck ( street , action . group ( ' PNAME ' ) )
2009-03-14 11:48:34 +01:00
elif action . group ( ' ATYPE ' ) == ' discards ' :
2009-07-11 19:44:32 +02:00
hand . addDiscard ( street , action . group ( ' PNAME ' ) , action . group ( ' BET ' ) , action . group ( ' DISCARDED ' ) )
2009-03-14 11:48:34 +01:00
elif action . group ( ' ATYPE ' ) == ' stands pat ' :
hand . addStandsPat ( street , action . group ( ' PNAME ' ) )
2009-03-06 02:24:38 +01:00
else :
2009-03-14 11:48:34 +01:00
print " DEBUG: unimplemented readAction: ' %s ' ' %s ' " % ( action . group ( ' PNAME ' ) , action . group ( ' ATYPE ' ) , )
2009-03-06 02:24:38 +01:00
def readShowdownActions ( self , hand ) :
2009-07-15 17:48:58 +02:00
# TODO: pick up mucks also??
2009-03-06 02:24:38 +01:00
for shows in self . re_ShowdownAction . finditer ( hand . handText ) :
2009-07-12 22:01:02 +02:00
cards = shows . group ( ' CARDS ' ) . split ( ' ' )
2009-03-06 02:24:38 +01:00
hand . addShownCards ( cards , shows . group ( ' PNAME ' ) )
def readCollectPot ( self , hand ) :
for m in self . re_CollectPot . finditer ( hand . handText ) :
hand . addCollectPot ( player = m . group ( ' PNAME ' ) , pot = m . group ( ' POT ' ) )
def readShownCards ( self , hand ) :
for m in self . re_ShownCards . finditer ( hand . handText ) :
if m . group ( ' CARDS ' ) is not None :
cards = m . group ( ' CARDS ' )
2009-07-04 20:35:20 +02:00
cards = cards . split ( ' ' ) # needs to be a list, not a set--stud needs the order
2009-07-04 00:59:50 +02:00
( shown , mucked ) = ( False , False )
if m . group ( ' SHOWED ' ) == " showed " : shown = True
elif m . group ( ' SHOWED ' ) == " mucked " : mucked = True
hand . addShownCards ( cards = cards , player = m . group ( ' PNAME ' ) , shown = shown , mucked = mucked )
2009-03-06 02:24:38 +01:00
if __name__ == " __main__ " :
parser = OptionParser ( )
2009-07-15 01:15:04 +02:00
parser . add_option ( " -i " , " --input " , dest = " ipath " , help = " parse input hand history " , default = " regression-test-files/stars/horse/HH20090226 Natalie V - $0.10-$0.20 - HORSE.txt " )
2009-03-06 02:24:38 +01:00
parser . add_option ( " -o " , " --output " , dest = " opath " , help = " output translation to " , default = " - " )
parser . add_option ( " -f " , " --follow " , dest = " follow " , help = " follow (tail -f) the input " , action = " store_true " , default = False )
2009-11-22 05:34:11 +01:00
#parser.add_option("-q", "--quiet", action="store_const", const=logging.CRITICAL, dest="verbosity", default=logging.INFO)
#parser.add_option("-v", "--verbose", action="store_const", const=logging.INFO, dest="verbosity")
#parser.add_option("--vv", action="store_const", const=logging.DEBUG, dest="verbosity")
2009-03-06 02:24:38 +01:00
( options , args ) = parser . parse_args ( )
e = PokerStars ( in_path = options . ipath , out_path = options . opath , follow = options . follow )