2010-07-08 20:01:03 +02:00
#!/usr/bin/python
2009-08-06 19:26:55 +02:00
# -*- coding: utf-8 -*-
2008-11-07 09:47:00 +01:00
2010-07-04 03:05:16 +02:00
#Copyright 2008-2010 Carl Gherardi
2008-11-07 09:47:00 +01:00
#This program is free software: you can redistribute it and/or modify
#it under the terms of the GNU Affero General Public License as published by
#the Free Software Foundation, version 3 of the License.
#
#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 Affero General Public License
#along with this program. If not, see <http://www.gnu.org/licenses/>.
2010-07-04 03:05:16 +02:00
#In the "official" distribution you can find the license in agpl-3.0.txt.
2008-11-07 09:47:00 +01:00
2008-11-09 12:57:58 +01:00
import re
2008-11-09 01:46:14 +01:00
import sys
import traceback
2009-02-26 01:59:36 +01:00
from optparse import OptionParser
2008-11-09 03:58:46 +01:00
import os
import os . path
2008-11-09 01:46:14 +01:00
import xml . dom . minidom
2008-12-10 17:30:57 +01:00
import codecs
2008-12-05 03:40:04 +01:00
from decimal import Decimal
import operator
2008-11-09 01:46:14 +01:00
from xml . dom . minidom import Node
2010-07-29 16:18:05 +02:00
2008-12-17 00:23:33 +01:00
import time
import datetime
2010-07-29 16:18:05 +02:00
from pytz import timezone
import pytz
2010-01-31 12:25:24 +01:00
2010-02-01 23:31:00 +01:00
import logging
# logging has been set up in fpdb.py or HUD_main.py, use their settings:
log = logging . getLogger ( " parser " )
2010-01-31 12:25:24 +01:00
import Hand
2009-08-09 16:19:43 +02:00
from Exceptions import FpdbParseError
2009-11-10 01:30:23 +01:00
import Configuration
2008-12-10 00:30:58 +01:00
2010-08-15 20:50:49 +02:00
import locale
lang = locale . getdefaultlocale ( ) [ 0 ] [ 0 : 2 ]
if lang == " en " :
def _ ( string ) : return string
else :
import gettext
try :
trans = gettext . translation ( " fpdb " , localedir = " locale " , languages = [ lang ] )
trans . install ( )
except IOError :
def _ ( string ) : return string
2009-08-09 16:19:43 +02:00
2009-09-10 05:10:55 +02:00
import pygtk
import gtk
2009-03-12 13:24:23 +01:00
class HandHistoryConverter ( ) :
2009-03-24 15:58:14 +01:00
2009-08-09 16:19:43 +02:00
READ_CHUNK_SIZE = 10000 # bytes to read at a time from file in tail mode
# filetype can be "text" or "xml"
# so far always "text"
# subclass HHC_xml for xml parsing
filetype = " text "
# codepage indicates the encoding of the text file.
# cp1252 is a safe default
# "utf_8" is more likely if there are funny characters
codepage = " cp1252 "
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
2010-04-23 05:49:57 +02:00
def __init__ ( self , config , in_path = ' - ' , out_path = ' - ' , follow = False , index = 0 , autostart = True , starsArchive = False , ftpArchive = False ) :
2009-08-09 16:19:43 +02:00
""" \
in_path ( default ' - ' = sys . stdin )
out_path ( default ' - ' = sys . stdout )
follow : whether to tail - f the input """
2010-01-31 12:25:24 +01:00
self . config = config
2010-02-19 23:50:45 +01:00
self . import_parameters = self . config . get_import_parameters ( )
2010-02-01 23:31:00 +01:00
#log = Configuration.get_logger("logging.conf", "parser", log_dir=self.config.dir_log)
2009-08-09 16:19:43 +02:00
log . info ( " HandHistory init - %s subclass, in_path ' %s ' ; out_path ' %s ' " % ( self . sitename , in_path , out_path ) )
2010-02-01 14:48:28 +01:00
2009-12-22 14:40:37 +01:00
self . index = index
2009-12-17 11:42:50 +01:00
self . starsArchive = starsArchive
2010-04-23 05:49:57 +02:00
self . ftpArchive = ftpArchive
2009-07-15 00:34:47 +02:00
2009-02-26 01:59:36 +01:00
self . in_path = in_path
self . out_path = out_path
2009-07-19 09:45:18 +02:00
self . processedHands = [ ]
2009-12-04 10:56:56 +01:00
self . numHands = 0
self . numErrors = 0
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
# Tourney object used to store TourneyInfo when called to deal with a Summary file
self . tourney = None
2010-02-01 14:48:28 +01:00
2009-07-15 00:34:47 +02:00
if in_path == ' - ' :
self . in_fh = sys . stdin
2010-02-19 23:50:45 +01:00
self . out_fh = get_out_fh ( out_path , self . import_parameters )
2009-07-15 00:34:47 +02:00
2009-02-26 01:59:36 +01:00
self . follow = follow
2009-03-02 00:22:47 +01:00
self . compiledPlayers = set ( )
2009-02-24 17:17:25 +01:00
self . maxseats = 10
2010-02-01 14:48:28 +01:00
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
self . status = True
self . parsedObjectType = " HH " #default behaviour : parsing HH files, can be "Summary" if the parsing encounters a Summary File
2008-12-06 15:15:41 +01:00
2009-08-09 16:19:43 +02:00
if autostart :
self . start ( )
2008-12-06 15:15:41 +01:00
def __str__ ( self ) :
2009-07-15 00:34:47 +02:00
return """
HandHistoryConverter : ' %(sitename)s '
2009-08-09 16:19:43 +02:00
filetype ' %(filetype)s '
in_path ' %(in_path)s '
out_path ' %(out_path)s '
follow ' %(follow)s '
2010-02-01 14:48:28 +01:00
""" % lo cals()
2008-12-06 15:15:41 +01:00
2009-03-12 13:24:23 +01:00
def start ( self ) :
2009-08-09 16:19:43 +02:00
""" Process a hand at a time from the input specified by in_path.
2009-03-10 00:03:17 +01:00
If in follow mode , wait for more data to turn up .
2009-08-09 16:19:43 +02:00
Otherwise , finish at EOF .
2009-04-02 22:20:41 +02:00
2009-07-15 06:22:46 +02:00
"""
2009-09-10 05:10:55 +02:00
while gtk . events_pending ( ) :
gtk . main_iteration ( False )
2009-03-10 22:49:23 +01:00
starttime = time . time ( )
if not self . sanityCheck ( ) :
2010-08-14 18:51:42 +02:00
log . warning ( _ ( " Failed sanity check " ) )
2009-03-10 22:49:23 +01:00
return
2009-03-12 16:31:29 +01:00
2009-08-09 16:19:43 +02:00
try :
2009-12-04 10:56:56 +01:00
self . numHands = 0
self . numErrors = 0
2009-08-09 16:19:43 +02:00
if self . follow :
2010-02-01 14:48:28 +01:00
#TODO: See how summary files can be handled on the fly (here they should be rejected as before)
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Tailing ' %s ' " ) % self . in_path )
2009-08-09 16:19:43 +02:00
for handText in self . tailHands ( ) :
try :
self . processHand ( handText )
2009-12-04 10:56:56 +01:00
self . numHands + = 1
2009-08-09 16:19:43 +02:00
except FpdbParseError , e :
2009-12-04 10:56:56 +01:00
self . numErrors + = 1
2010-08-14 18:51:42 +02:00
log . warning ( _ ( " HHC.start(follow): processHand failed: Exception msg: ' %s ' " ) % e )
2009-08-09 16:19:43 +02:00
log . debug ( handText )
else :
handsList = self . allHandsAsList ( )
log . info ( " Parsing %d hands " % len ( handsList ) )
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
# Determine if we're dealing with a HH file or a Summary file
2009-09-04 02:01:41 +02:00
# quick fix : empty files make the handsList[0] fail ==> If empty file, go on with HH parsing
if len ( handsList ) == 0 or self . isSummary ( handsList [ 0 ] ) == False :
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
self . parsedObjectType = " HH "
for handText in handsList :
try :
self . processedHands . append ( self . processHand ( handText ) )
except FpdbParseError , e :
2009-12-04 10:56:56 +01:00
self . numErrors + = 1
2010-08-14 18:51:42 +02:00
log . warning ( _ ( " HHC.start(): processHand failed: Exception msg: ' %s ' " ) % e )
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
log . debug ( handText )
2009-12-04 10:56:56 +01:00
self . numHands = len ( handsList )
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
endtime = time . time ( )
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Read %d hands ( %d failed) in %.3f seconds " ) % ( self . numHands , self . numErrors , endtime - starttime ) )
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
else :
self . parsedObjectType = " Summary "
summaryParsingStatus = self . readSummaryInfo ( handsList )
endtime = time . time ( )
if summaryParsingStatus :
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Summary file ' %s ' correctly parsed (took %.3f seconds) " ) % ( self . in_path , endtime - starttime ) )
2010-02-01 14:48:28 +01:00
else :
2010-08-14 18:51:42 +02:00
log . warning ( _ ( " Error converting summary file ' %s ' (took %.3f seconds) " ) % ( self . in_path , endtime - starttime ) )
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
2009-08-09 16:19:43 +02:00
except IOError , ioe :
2010-08-14 18:51:42 +02:00
log . exception ( _ ( " Error converting ' %s ' " ) % self . in_path )
2009-08-09 16:19:43 +02:00
finally :
if self . out_fh != sys . stdout :
self . out_fh . close ( )
2009-03-12 16:31:29 +01:00
2009-02-26 01:59:36 +01:00
def tailHands ( self ) :
2009-03-10 17:17:54 +01:00
""" Generator of handTexts from a tailed file:
2009-04-02 22:20:41 +02:00
Tail the in_path file and yield handTexts separated by re_SplitHands .
This requires a regex that greedily groups and matches the ' splitter ' between hands ,
which it expects to find at self . re_TailSplitHands - - see for e . g . Everleaf . py .
"""
2009-11-03 21:14:20 +01:00
if self . in_path == ' - ' :
raise StopIteration
2009-03-10 17:17:54 +01:00
interval = 1.0 # seconds to sleep between reads for new data
2009-03-12 13:24:23 +01:00
fd = codecs . open ( self . in_path , ' r ' , self . codepage )
2009-03-10 17:17:54 +01:00
data = ' '
while 1 :
where = fd . tell ( )
newdata = fd . read ( self . READ_CHUNK_SIZE )
if not newdata :
fd_results = os . fstat ( fd . fileno ( ) )
try :
2009-03-12 13:24:23 +01:00
st_results = os . stat ( self . in_path )
2009-03-10 17:17:54 +01:00
except OSError :
st_results = fd_results
if st_results [ 1 ] == fd_results [ 1 ] :
time . sleep ( interval )
fd . seek ( where )
else :
2010-08-14 18:51:42 +02:00
log . debug ( _ ( " %s changed inode numbers from %d to %d " ) % ( self . in_path , fd_results [ 1 ] , st_results [ 1 ] ) )
2009-03-12 13:24:23 +01:00
fd = codecs . open ( self . in_path , ' r ' , self . codepage )
2009-03-10 17:17:54 +01:00
fd . seek ( where )
2009-02-26 01:59:36 +01:00
else :
2009-03-10 17:17:54 +01:00
# yield hands
data = data + newdata
2009-04-02 22:20:41 +02:00
result = self . re_TailSplitHands . split ( data )
2009-03-10 17:17:54 +01:00
result = iter ( result )
2009-03-24 15:58:14 +01:00
data = ' '
2009-03-10 17:17:54 +01:00
# --x data (- is bit of splitter, x is paragraph) yield,...,keep
# [,--,x] result of re.split (with group around splitter)
# ,x our output: yield nothing, keep x
#
# --x--x [,--,x,--,x] x,x
# -x--x [-x,--,x] x,x
# x- [x-] ,x-
# x-- [x,--,] x,--
# x--x [x,--,x] x,x
# x--x-- [x,--,x,--,] x,x,--
2010-02-01 14:48:28 +01:00
2009-03-10 17:17:54 +01:00
# The length is always odd.
# 'odd' indices are always splitters.
# 'even' indices are always paragraphs or ''
# We want to discard all the ''
# We want to discard splitters unless the final item is '' (because the splitter could grow with new data)
# We want to yield all paragraphs followed by a splitter, i.e. all even indices except the last.
for para in result :
try :
2009-03-24 15:58:14 +01:00
result . next ( )
splitter = True
2009-03-10 17:17:54 +01:00
except StopIteration :
2009-03-24 15:58:14 +01:00
splitter = False
2009-03-10 17:17:54 +01:00
if splitter : # para is followed by a splitter
if para : yield para # para not ''
else :
data = para # keep final partial paragraph
def allHandsAsList ( self ) :
2009-02-26 01:59:36 +01:00
""" Return a list of handtexts in the file at self.in_path """
2009-03-10 22:49:23 +01:00
#TODO : any need for this to be generator? e.g. stars support can email one huge file of all hands in a year. Better to read bit by bit than all at once.
2009-02-26 01:59:36 +01:00
self . readFile ( )
self . obs = self . obs . strip ( )
self . obs = self . obs . replace ( ' \r \n ' , ' \n ' )
2009-12-17 11:42:50 +01:00
if self . starsArchive == True :
2010-08-14 18:51:42 +02:00
log . debug ( _ ( " Converting starsArchive format to readable " ) )
2009-12-17 11:42:50 +01:00
m = re . compile ( ' ^Hand # \ d+ ' , re . MULTILINE )
self . obs = m . sub ( ' ' , self . obs )
2010-04-23 05:49:57 +02:00
if self . ftpArchive == True :
2010-08-14 18:51:42 +02:00
log . debug ( _ ( " Converting ftpArchive format to readable " ) )
2010-04-23 05:49:57 +02:00
m = re . compile ( ' ^ \ * \ * \ * \ * \ * \ *+ \ s# \ s \ d+ \ s \ * \ * \ * \ * \ *+$ ' , re . MULTILINE )
self . obs = m . sub ( ' ' , self . obs )
2009-11-03 20:30:52 +01:00
if self . obs is None or self . obs == " " :
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Read no hands. " ) )
2009-08-12 10:47:18 +02:00
return [ ]
2009-04-02 22:16:36 +02:00
return re . split ( self . re_SplitHands , self . obs )
2010-02-01 14:48:28 +01:00
2009-03-10 17:17:54 +01:00
def processHand ( self , handText ) :
gametype = self . determineGameType ( handText )
2009-08-09 16:19:43 +02:00
log . debug ( " gametype %s " % gametype )
2009-07-16 06:58:39 +02:00
hand = None
2009-11-29 09:54:15 +01:00
l = None
2010-02-01 14:48:28 +01:00
if gametype is None :
2009-03-14 16:02:23 +01:00
gametype = " unmatched "
2010-08-18 07:17:42 +02:00
# TODO: not ideal, just trying to not error. Throw ParseException?
self . numErrors + = 1
2009-03-14 16:02:23 +01:00
else :
# See if gametype is supported.
type = gametype [ ' type ' ]
base = gametype [ ' base ' ]
limit = gametype [ ' limitType ' ]
l = [ type ] + [ base ] + [ limit ]
2009-03-12 11:43:28 +01:00
if l in self . readSupportedGames ( ) :
if gametype [ ' base ' ] == ' hold ' :
2009-08-09 16:19:43 +02:00
log . debug ( " hand = Hand.HoldemOmahaHand(self, self.sitename, gametype, handtext) " )
2010-01-31 12:25:24 +01:00
hand = Hand . HoldemOmahaHand ( self . config , self , self . sitename , gametype , handText )
2009-03-12 11:43:28 +01:00
elif gametype [ ' base ' ] == ' stud ' :
2010-01-31 12:25:24 +01:00
hand = Hand . StudHand ( self . config , self , self . sitename , gametype , handText )
2009-03-12 11:43:28 +01:00
elif gametype [ ' base ' ] == ' draw ' :
2010-01-31 12:25:24 +01:00
hand = Hand . DrawHand ( self . config , self , self . sitename , gametype , handText )
2009-03-12 11:43:28 +01:00
else :
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Unsupported game type: %s " % gametype ) )
2009-03-10 00:03:17 +01:00
2009-03-05 02:12:15 +01:00
if hand :
2010-01-23 06:37:41 +01:00
#hand.writeHand(self.out_fh)
2009-07-19 09:45:18 +02:00
return hand
2009-03-05 02:12:15 +01:00
else :
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Unsupported game type: %s " % gametype ) )
2009-03-05 02:12:15 +01:00
# TODO: pity we don't know the HID at this stage. Log the entire hand?
# From the log we can deduce that it is the hand after the one before :)
2008-12-10 17:30:57 +01:00
2009-03-12 11:43:28 +01:00
2008-12-06 15:15:41 +01:00
# These functions are parse actions that may be overridden by the inheriting class
2009-02-05 10:50:50 +01:00
# This function should return a list of lists looking like:
# return [["ring", "hold", "nl"], ["tour", "hold", "nl"]]
# Showing all supported games limits and types
2010-02-01 14:48:28 +01:00
2008-12-06 15:15:41 +01:00
def readSupportedGames ( self ) : abstract
# should return a list
# type base limit
# [ ring, hold, nl , sb, bb ]
# Valid types specified in docs/tabledesign.html in Gametypes
2009-03-05 16:17:17 +01:00
def determineGameType ( self , handText ) : abstract
2009-03-10 00:03:17 +01:00
""" return dict with keys/values:
' type ' in ( ' ring ' , ' tour ' )
' limitType ' in ( ' nl ' , ' cn ' , ' pl ' , ' cp ' , ' fl ' )
' base ' in ( ' hold ' , ' stud ' , ' draw ' )
' category ' in ( ' holdem ' , ' omahahi ' , omahahilo ' , ' razz ' , ' studhi ' , ' studhilo ' , ' fivedraw ' , ' 27_1 draw ' , ' 27_3 draw ' , ' badugi ' )
' hilo ' in ( ' h ' , ' l ' , ' s ' )
' smallBlind ' int ?
' bigBlind ' int ?
' smallBet '
' bigBet '
' currency ' in ( ' USD ' , ' EUR ' , ' T$ ' , < countrycode > )
or None if we fail to get the info """
#TODO: which parts are optional/required?
2008-12-06 15:15:41 +01:00
def readHandInfo ( self , hand ) : abstract
2010-08-20 12:52:00 +02:00
""" Read and set information about the hand being dealt, and set the correct
variables in the Hand object ' hand
* hand . startTime - a datetime object
* hand . handid - The site identified for the hand - a string .
* hand . tablename
* hand . buttonpos
* hand . maxseats
* hand . mixed
Tournament fields :
* hand . tourNo - The site identified tournament id as appropriate - a string .
* hand . buyin
* hand . fee
* hand . buyinCurrency
* hand . koBounty
* hand . isKO
* hand . level
"""
#TODO: which parts are optional/required?
2008-12-06 15:15:41 +01:00
# Needs to return a list of lists in the format
# [['seat#', 'player1name', 'stacksize'] ['seat#', 'player2name', 'stacksize'] [...]]
def readPlayerStacks ( self , hand ) : abstract
2010-02-01 14:48:28 +01:00
2009-02-25 15:35:28 +01:00
def compilePlayerRegexs ( self ) : abstract
""" Compile dynamic regexes -- these explicitly match known player names and must be updated if a new player joins """
2010-02-01 14:48:28 +01:00
2008-12-06 15:15:41 +01:00
# Needs to return a MatchObject with group names identifying the streets into the Hand object
2008-12-14 23:05:51 +01:00
# so groups are called by street names 'PREFLOP', 'FLOP', 'STREET2' etc
# blinds are done seperately
2008-12-06 15:15:41 +01:00
def markStreets ( self , hand ) : abstract
#Needs to return a list in the format
2010-02-01 14:48:28 +01:00
# ['player1name', 'player2name', ...] where player1name is the sb and player2name is bb,
2008-12-06 15:15:41 +01:00
# addtional players are assumed to post a bb oop
def readBlinds ( self , hand ) : abstract
2009-02-24 17:17:25 +01:00
def readAntes ( self , hand ) : abstract
def readBringIn ( self , hand ) : abstract
def readButton ( self , hand ) : abstract
2008-12-06 15:15:41 +01:00
def readHeroCards ( self , hand ) : abstract
2009-02-25 11:32:12 +01:00
def readPlayerCards ( self , hand , street ) : abstract
2008-12-06 15:15:41 +01:00
def readAction ( self , hand , street ) : abstract
2008-12-09 16:32:37 +01:00
def readCollectPot ( self , hand ) : abstract
2008-12-16 18:14:37 +01:00
def readShownCards ( self , hand ) : abstract
2009-07-22 16:24:29 +02:00
# Some sites do odd stuff that doesn't fall in to the normal HH parsing.
2010-02-01 14:48:28 +01:00
# e.g., FTP doesn't put mixed game info in the HH, but puts in in the
2009-07-22 16:24:29 +02:00
# file name. Use readOther() to clean up those messes.
def readOther ( self , hand ) : pass
2010-02-01 14:48:28 +01:00
2008-12-06 15:15:41 +01:00
# Some sites don't report the rake. This will be called at the end of the hand after the pot total has been calculated
2008-12-14 23:05:51 +01:00
# an inheriting class can calculate it for the specific site if need be.
def getRake ( self , hand ) :
hand . rake = hand . totalpot - hand . totalcollected # * Decimal('0.05') # probably not quite right
2010-02-01 14:48:28 +01:00
2008-12-06 15:15:41 +01:00
def sanityCheck ( self ) :
2009-03-10 22:49:23 +01:00
""" Check we aren ' t going to do some stupid things """
#TODO: the hhbase stuff needs to be in fpdb_import
2008-12-14 20:25:04 +01:00
sane = False
2008-12-06 15:15:41 +01:00
base_w = False
2009-03-10 22:49:23 +01:00
#~ #Check if hhbase exists and is writable
#~ #Note: Will not try to create the base HH directory
#~ if not (os.access(self.hhbase, os.W_OK) and os.path.isdir(self.hhbase)):
#~ print "HH Sanity Check: Directory hhbase '" + self.hhbase + "' doesn't exist or is not writable"
#~ else:
#~ #Check if hhdir exists and is writable
#~ if not os.path.isdir(self.hhdir):
#~ # In first pass, dir may not exist. Attempt to create dir
#~ print "Creating directory: '%s'" % (self.hhdir)
#~ os.mkdir(self.hhdir)
#~ sane = True
#~ elif os.access(self.hhdir, os.W_OK):
#~ sane = True
#~ else:
#~ print "HH Sanity Check: Directory hhdir '" + self.hhdir + "' or its parent directory are not writable"
2008-12-06 15:15:41 +01:00
2009-02-21 14:06:10 +01:00
# Make sure input and output files are different or we'll overwrite the source file
2009-03-10 22:49:23 +01:00
if True : # basically.. I don't know
sane = True
2010-02-01 14:48:28 +01:00
2009-11-03 21:14:20 +01:00
if self . in_path != ' - ' and self . out_path == self . in_path :
2010-08-14 18:51:42 +02:00
print _ ( " HH Sanity Check: output and input files are the same, check config " )
2009-03-10 22:49:23 +01:00
sane = False
2009-02-21 14:06:10 +01:00
2008-12-06 15:15:41 +01:00
return sane
# Functions not necessary to implement in sub class
2008-12-10 17:30:57 +01:00
def setFileType ( self , filetype = " text " , codepage = ' utf8 ' ) :
2008-12-06 15:15:41 +01:00
self . filetype = filetype
2008-12-10 17:30:57 +01:00
self . codepage = codepage
2008-12-06 15:15:41 +01:00
2009-07-19 09:45:18 +02:00
#This function doesn't appear to be used
2008-12-06 15:15:41 +01:00
def splitFileIntoHands ( self ) :
hands = [ ]
2009-02-26 01:59:36 +01:00
self . obs = self . obs . strip ( )
2009-02-20 17:29:52 +01:00
list = self . re_SplitHands . split ( self . obs )
2008-12-06 15:15:41 +01:00
list . pop ( ) #Last entry is empty
for l in list :
2009-03-04 16:10:08 +01:00
# print "'" + l + "'"
2010-01-31 12:25:24 +01:00
hands = hands + [ Hand . Hand ( self . config , self . sitename , self . gametype , l ) ]
2009-11-03 21:14:20 +01:00
# TODO: This looks like it could be replaced with a list comp.. ?
2008-12-06 15:15:41 +01:00
return hands
2009-08-24 01:40:39 +02:00
def __listof ( self , x ) :
2009-11-03 21:14:20 +01:00
if isinstance ( x , list ) or isinstance ( x , tuple ) :
return x
else :
return [ x ]
2009-08-24 01:40:39 +02:00
2009-02-26 01:59:36 +01:00
def readFile ( self ) :
2009-08-09 16:19:43 +02:00
""" Open in_path according to self.codepage. Exceptions caught further up """
2010-02-01 14:48:28 +01:00
2009-11-03 21:14:20 +01:00
if self . filetype == " text " :
2009-02-26 01:59:36 +01:00
if self . in_path == ' - ' :
# read from stdin
2010-08-14 18:51:42 +02:00
log . debug ( _ ( " Reading stdin with %s " ) % self . codepage ) # is this necessary? or possible? or what?
2009-02-26 01:59:36 +01:00
in_fh = codecs . getreader ( ' cp1252 ' ) ( sys . stdin )
else :
2009-08-24 01:40:39 +02:00
for kodec in self . __listof ( self . codepage ) :
2009-09-11 07:38:20 +02:00
#print "trying", kodec
2009-08-24 01:40:39 +02:00
try :
in_fh = codecs . open ( self . in_path , ' r ' , kodec )
2010-01-26 16:56:07 +01:00
whole_file = in_fh . read ( )
2010-08-03 12:24:03 +02:00
in_fh . close ( )
2010-01-26 16:56:07 +01:00
self . obs = whole_file [ self . index : ]
self . index = len ( whole_file )
2009-09-11 07:38:20 +02:00
break
2009-08-24 01:40:39 +02:00
except :
pass
2009-09-11 07:38:20 +02:00
else :
2010-08-14 18:51:42 +02:00
print _ ( " unable to read file with any codec in list! " ) , self . in_path
2010-08-05 18:31:37 +02:00
self . obs = " "
2009-11-03 21:14:20 +01:00
elif self . filetype == " xml " :
2009-08-09 16:19:43 +02:00
doc = xml . dom . minidom . parse ( filename )
self . doc = doc
2008-12-06 15:15:41 +01:00
2009-07-20 16:01:51 +02:00
def guessMaxSeats ( self , hand ) :
2009-11-01 02:25:27 +01:00
""" Return a guess at maxseats when not specified in HH. """
# if some other code prior to this has already set it, return it
2010-02-20 02:14:34 +01:00
if self . maxseats > 1 and self . maxseats < 11 :
return self . maxseats
2009-07-20 16:01:51 +02:00
mo = self . maxOccSeat ( hand )
if mo == 10 : return 10 #that was easy
if hand . gametype [ ' base ' ] == ' stud ' :
if mo < = 8 : return 8
2010-02-01 14:48:28 +01:00
else : return mo
2009-07-20 16:01:51 +02:00
if hand . gametype [ ' base ' ] == ' draw ' :
if mo < = 6 : return 6
else : return mo
if mo == 2 : return 2
if mo < = 6 : return 6
return 10
def maxOccSeat ( self , hand ) :
max = 0
for player in hand . players :
2009-11-03 21:14:20 +01:00
if player [ 0 ] > max :
max = player [ 0 ]
2009-07-20 16:01:51 +02:00
return max
2008-11-07 11:19:18 +01:00
2009-02-05 10:28:18 +01:00
def getStatus ( self ) :
#TODO: Return a status of true if file processed ok
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
return self . status
2009-02-05 10:28:18 +01:00
2009-07-19 09:45:18 +02:00
def getProcessedHands ( self ) :
return self . processedHands
2009-02-05 10:28:18 +01:00
def getProcessedFile ( self ) :
2009-03-10 22:49:23 +01:00
return self . out_path
2009-07-17 11:45:22 +02:00
def getLastCharacterRead ( self ) :
return self . index
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
def isSummary ( self , topline ) :
return " Tournament Summary " in topline
def getParsedObjectType ( self ) :
return self . parsedObjectType
2010-02-01 14:48:28 +01:00
#returns a status (True/False) indicating wether the parsing could be done correctly or not
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
def readSummaryInfo ( self , summaryInfoList ) : abstract
2010-02-01 14:48:28 +01:00
First attempt to parsing summary files from FTP
modified: FulltiltToFpdb.py
* Modified re_HandInfo regex to take into account Matrix Tournament Hands Histories
* Add Regex for Tourney Summaries files parsing
* New methods : readSummaryInfo, determineTourneyType, getPlayersPositionsAndWinnings
modified: HandHistoryConverter.py
* Add a Tourney object in the attributes to allow storing the summary info retrieved and the ability to hand it over after parsing
* Add a new attribut parsedObjectType (string : "HH" or "Summary")
* In follow = False mode : read the first line to swicth between HH & Summary parsing
* TO DO : Deal with parsing summary files in follow = True mode
* New methods added : isSummary, getParsedObjectType (to be called in fpdb_import), readSummaryInfo : abstract (to be implemented in each specific HHC), getTourney (returns the new attribute)
new file: Tourney.py
* New object
* Lots of attributes regarding the tourney info (buy-in, fee, entries, speed, Rebuy/add-on info, starting chips, KnockOut info, isHeadsUp, isShootout, isMatrix, ...
* List of players with finishing positions (when available) and winnings (when available)
* Methods :
** addPlayer(self, rank, name, winnings)
** incrementPlayerWinnings(self, name, additionnalWinnings): used for KO tourneys when KO occured (for Hero only)
** calculatePayinAmount : unused yet, should be necessary for DB storage
** some Hand methods copied that are still to be done including : assemble and insert
** TO DO : write the insert method for the object to be stored in DB. This will have to deal with the fact that the DB write can occur both before (Bulk Import) or after the HH file info has been stored (tourney might or might not already exist)
modified: fpdb_import.py
* import_file_dict modified :
** after the construction of the hhc, it now calls the new getParsedObjectType method of HHC in order to know what has been parsed by the HHC
** If it's a hand history file (actual expected behaviour) : do as before
** If it's a summary file, gets the tourney object that has been built from the hhc and calls the insert method on it (similar to NEWIMPORT=True for HH)
2009-08-22 00:09:34 +02:00
def getTourney ( self ) :
return self . tourney
2010-07-13 20:37:06 +02:00
@staticmethod
def changeTimezone ( time , givenTimezone , wantedTimezone ) :
2010-07-29 16:18:05 +02:00
#print "raw time:",time, "given TZ:", givenTimezone
if wantedTimezone == " UTC " :
wantedTimezone = pytz . utc
else :
raise Error #TODO raise appropriate error
if givenTimezone == " ET " :
givenTimezone = timezone ( ' US/Eastern ' )
elif givenTimezone == " CET " :
givenTimezone = timezone ( ' Europe/Berlin ' )
#Note: Daylight Saving Time is standardised across the EU so this should be fine
2010-07-29 16:57:57 +02:00
elif givenTimezone == ' HST ' : # Hawaiian Standard Time
pass
elif givenTimezone == ' AKT ' : # Alaska Time
pass
elif givenTimezone == ' PT ' : # Pacific Time
pass
elif givenTimezone == ' MT ' : # Mountain Time
pass
elif givenTimezone == ' CT ' : # Central Time
pass
elif givenTimezone == ' AT ' : # Atlantic Time
pass
elif givenTimezone == ' NT ' : # Newfoundland Time
pass
elif givenTimezone == ' ART ' : # Argentinian Time
pass
elif givenTimezone == ' BRT ' : # Brasilia Time
pass
elif givenTimezone == ' AKT ' : # Alaska Time
pass
elif givenTimezone == ' WET ' : # Western European Time
pass
elif givenTimezone == ' EET ' : # Eastern European Time
pass
elif givenTimezone == ' MSK ' : # Moscow Standard Time
pass
elif givenTimezone == ' IST ' : # India Standard Time
pass
elif givenTimezone == ' CCT ' : # China Coast Time
pass
elif givenTimezone == ' JST ' : # Japan Standard Time
pass
elif givenTimezone == ' AWST ' : # Australian Western Standard Time
givenTimezone = timezone ( ' Australia/West ' )
elif givenTimezone == ' ACST ' : # Australian Central Standard Time
givenTimezone = timezone ( ' Australia/Darwin ' )
elif givenTimezone == ' AEST ' : # Australian Eastern Standard Time
# Each State on the East Coast has different DSTs.
# Melbournce is out because I don't like AFL, Queensland doesn't have DST
# ACT is full of politicians and Tasmania will never notice.
# Using Sydney.
givenTimezone = timezone ( ' Australia/Sydney ' )
elif givenTimezone == ' NZT ' : # New Zealand Time
pass
2010-07-29 16:18:05 +02:00
else :
raise Error #TODO raise appropriate error
localisedTime = givenTimezone . localize ( time )
utcTime = localisedTime . astimezone ( wantedTimezone )
#print "utcTime:",utcTime
return utcTime
2010-07-13 20:37:06 +02:00
#end @staticmethod def changeTimezone
2009-11-07 18:30:47 +01:00
@staticmethod
def getTableTitleRe ( type , table_name = None , tournament = None , table_number = None ) :
" Returns string to search in windows titles "
if type == " tour " :
2009-12-23 21:12:56 +01:00
return " %s .+Table.+ %s " % ( tournament , table_number )
2009-11-07 18:30:47 +01:00
else :
return table_name
def getTableTitleRe ( config , sitename , * args , * * kwargs ) :
" Returns string to search in windows titles for current site "
return getSiteHhc ( config , sitename ) . getTableTitleRe ( * args , * * kwargs )
def getSiteHhc ( config , sitename ) :
" Returns HHC class for current site "
hhcName = config . supported_sites [ sitename ] . converter
hhcModule = __import__ ( hhcName )
return getattr ( hhcModule , hhcName [ : - 6 ] )
2010-02-19 23:50:45 +01:00
def get_out_fh ( out_path , parameters ) :
if out_path == ' - ' :
return ( sys . stdout )
elif parameters [ ' saveStarsHH ' ] :
2010-02-20 02:14:34 +01:00
out_dir = os . path . dirname ( out_path )
2010-02-19 23:50:45 +01:00
if not os . path . isdir ( out_dir ) and out_dir != ' ' :
try :
os . makedirs ( out_dir )
except : # we get a WindowsError here in Windows.. pretty sure something else for Linux :D
2010-08-14 18:51:42 +02:00
log . error ( _ ( " Unable to create output directory %s for HHC! " ) % out_dir )
print _ ( " *** ERROR: UNABLE TO CREATE OUTPUT DIRECTORY " ) , out_dir
2010-02-19 23:50:45 +01:00
else :
2010-08-14 18:51:42 +02:00
log . info ( _ ( " Created directory ' %s ' " ) % out_dir )
2010-02-19 23:50:45 +01:00
try :
2010-02-20 02:14:34 +01:00
return ( codecs . open ( out_path , ' w ' , ' utf8 ' ) )
2010-02-19 23:50:45 +01:00
except :
2010-08-14 18:51:42 +02:00
log . error ( _ ( " out_path %s couldn ' t be opened " ) % ( out_path ) )
2010-02-19 23:50:45 +01:00
else :
2010-02-20 02:14:34 +01:00
return ( sys . stdout )