#format python """simpocker.py Pocker simulation for probability varaition by user number By Hyung-Yong Kim, 2003-1 http://bioinfo.sarang.net/wiki/PockerSimulation """ import unittest, random ## item sequence for ordering rank SEQ_NUMBER=('2','3','4','5','6','7','8','9','0','J','Q','K','1') SEQ_NUMBER_REVERSE=('1','K','Q','J','0','9','8','7','6','5','4','3','2','1') SEQ_FEATURE=('C','H','D','S') SEQ_FEATURE_REVERSE=('S','D','H','C') SEQ_CARDS=('StiFl','Pocker','House','Flush','Straight','Triple', 'TwoPair','OnePair') class Trump: def __init__(self): self.cards=[] for number in SEQ_NUMBER: for feature in SEQ_FEATURE: self.cards.append(number+feature) random.shuffle(self.cards) def getCard(self): card=self.cards.pop() return card class Cards: def __init__(self, aCardList=None): for each in SEQ_CARDS: setattr(self, each, None) self.Triple=[]; self.OnePair=[] if aCardList: self.cards=aCardList self.initdecide() else: self.cards=[] def put(self, aCard): self.cards.append(aCard) if len(self.cards) == 7: self.initdecide() def get(self): return self.cards def __str__(self): return str(self.cards) def initdecide(self): self.numberOfFeatures = {'S':0, 'D':0, 'H':0, 'C':0} self.numberOfNumbers = {'0':0, '1':0, '2':0, '3':0, '4':0, '5':0, '6':0, '7':0, '8':0, '9':0, 'J':0, 'Q':0, 'K':0,} self.numbers=[] for card in self.cards: self.numbers.append(card[0]) self.numberOfFeatures[card[1]]+=1 self.numberOfNumbers[card[0]]+=1 for fName in SEQ_FEATURE: if self.numberOfFeatures[fName] >= 5: self.Flush=fName for nName in SEQ_NUMBER: if self.numberOfNumbers[nName] == 4: self.Pocker=nName elif self.numberOfNumbers[nName] == 3: self.Triple.append(nName) elif self.numberOfNumbers[nName] == 2: self.OnePair.append(nName) def findCardByNum(self, aNum): if not aNum: return for fName in SEQ_FEATURE_REVERSE: card = aNum + fName if card in self.cards: return card def findCardByFea(self, aFea): if not aFea: return for nName in SEQ_NUMBER_REVERSE: card = nName + aFea if card in self.cards: return card def isFlush(self): return self.findCardByFea(self.Flush) def isPocker(self): return self.findCardByNum(self.Pocker) def isHouse(self): if len(self.Triple) ==2: return self.Triple elif self.Triple and self.OnePair: return [self.OnePair[0], self.Triple[0]] def isTriple(self): if self.Triple: return self.Triple[-1] def isOnePair(self): if self.OnePair: return self.findCardByNum(self.OnePair[-1]) def isTwoPair(self): if self.OnePair: if len(self.OnePair) > 1: return [self.findCardByNum(self.OnePair[-2]), self.findCardByNum(self.OnePair[-1]), ] def isStraight(self,aNumberList=None): if aNumberList: aList = aNumberList else: aList = self.numbers i=0;j=0 for num in SEQ_NUMBER_REVERSE: j+=1 if num in aList: i+=1 if i==5: self.Straight = SEQ_NUMBER_REVERSE[j-5] break continue i=0 return self.findCardByNum(self.Straight) def isStiFl(self): if self.isFlush(): feature = self.isFlush()[1] newList=[] for card in self.cards: if card[1] == feature: newList.append(card[0]) return self.isStraight(newList) def decide(self): if self.isStiFl(): return ('StiFl',self.isStiFl()) elif self.isPocker(): return ('Pocker',self.isPocker()) elif self.isHouse(): return ('House',self.isHouse()) elif self.isFlush(): return ('Flush',self.isFlush()) elif self.isStraight(): return ('Straight',self.isStraight()) elif self.isTriple(): return ('Triple',self.isTriple()) elif self.isTwoPair(): return ('TwoPair',self.isTwoPair()) elif self.isOnePair(): return ('OnePair',self.isOnePair()) else: for num in SEQ_NUMBER_REVERSE: if num in self.numbers: for feature in SEQ_FEATURE_REVERSE: card = num+feature if card in self.cards: return (None, card) class Game: def __init__(self, aNumOfPlayers): t=Trump() self.results=[] for player in range(aNumOfPlayers): c=Cards() for i in range(7): c.put(t.getCard()) self.results.append(c) def getRawResult(self): return self.results def getDecidedResult(self): decidedResult=[] for cards in self.results: decidedResult.append(cards.decide()) return decidedResult def getSimpleJokboList(self): return getEachItems(self.getDecidedResult(),0) def judgeWinner(self): return judge(self.getDecidedResult()) def getResultRecord(self): #record4=(jwinnerIndex,jokbo1,jokbo2,jokbo3,jokbo4,...) record=[] record.append(self.judgeWinner()[0]) record.extend(self.getSimpleJokboList()) return record class GameStat: def __init__(self, aNumOfGames, aNumOfPlayers): self.numOfGames = aNumOfGames self.numOfPlayers = aNumOfPlayers self.result=[] for i in range(aNumOfGames): g=Game(aNumOfPlayers) self.result.append(g.getResultRecord()) self.jokboFreq={}; self.jokboFreq[None]=0 for cards in SEQ_CARDS: self.jokboFreq[cards]=0 self.jokboWin = self.jokboFreq.copy() def getRawResult(self): return self.result def calculateEmitProb(self): for eachRecord in self.result: for jokbo in eachRecord[1:]: self.jokboFreq[jokbo]+=1 def getEmitProb(self): n = self.numOfGames * self.numOfPlayers return self.jokboFreq, n def calculateWinProb(self): for eachRecord in self.result: win=eachRecord[eachRecord[0]+1] self.jokboWin[win]+=1 for jokbo in eachRecord[1:]: self.jokboFreq[jokbo]+=1 def getWinProb(self): return self.jokboWin, self.jokboFreq def getEachItems(aList,aIth): """return i'th items in list """ result=[] for each in aList: result.append(each[aIth]) return result def getHighestCard(aList): winNum=None; winFea=None if len(aList[0]) == 1: numList = aList else: numList = getEachItems(aList,0) for num in SEQ_NUMBER_REVERSE: if num in numList: winNum=num if len(aList[0]) == 1: return num ## if house, return only num break lastList=[] for card in aList: if card[0]==winNum: lastList.append(card) for fea in SEQ_FEATURE_REVERSE: if fea in getEachItems(lastList,1): winFea=fea break return winNum+winFea def judge(aList): cardsNames = getEachItems(aList,0) highCardList = getEachItems(aList,1) winnerIndex=None; winners=[]; winnerJokbo=None for jokbo in SEQ_CARDS: if jokbo in cardsNames: winnerJokbo=jokbo if cardsNames.count(jokbo) ==1: winnerIndex=cardsNames.index(jokbo) return (winnerIndex, winnerJokbo) else: i=0 for myJokbo in cardsNames: if jokbo == myJokbo: winners.append(i) i+=1 lastCardList=[];lastCardAll=[] for j in winners: if jokbo in ('TwoPair','House'): lastCard=highCardList[j][1] lastCardAll.append(highCardList[j]) else: lastCard=highCardList[j] lastCardList.append(lastCard) hgCard=getHighestCard(lastCardList) if jokbo in ('TwoPair','House'): for oneCards in lastCardAll: if hgCard in oneCards: hgCardIndex=highCardList.index(oneCards) else: hgCardIndex=highCardList.index(hgCard) return (hgCardIndex, winnerJokbo) ## if all None hgCard=getHighestCard(highCardList) return (highCardList.index(hgCard), None) ### UnitTests #################################################### class PockerTest(unittest.TestCase): def testTrump(self): t=Trump() self.assertEquals(52, len(t.cards)) t.getCard() self.assertEquals(51, len(t.cards)) def testCards(self): c=Cards(['1S','2S','3D','4S','2C','9H','0H']) self.assertEquals(['1S','2S','3D','4S','2C','9H','0H'], c.get()) def testGame(self): g=Game(4) result=g.getRawResult() self.assertEquals(4, len(result)) for card in result: self.assertEquals(7,len(card.cards)) self.assertNotEquals(result[0], result[1]) class CardsEachJudgmentTest(unittest.TestCase): def testFlush(self): c=Cards(['1S', '2S', '3S', '4S', '5S', '7C', '8C']) self.assertEquals('1S',c.isFlush()) c=Cards(['1S', '2S', '3S', '4S', '5C', '7C', '8C']) self.assertEquals(None, c.isFlush()) c=Cards(['1S', '2S', '3C', '4C', '5C', '7C', '8C']) self.assertEquals('8C', c.isFlush()) def testPocker(self): c=Cards(['1S','1D','1H','1C','2S','4S','6C']) self.assertEquals('1S', c.isPocker()) def testHouse(self): c=Cards(['1S','1D','1H','2S','2D','2H','6C']) self.assertEquals(['2','1'], c.isHouse()) c=Cards(['1S','1D','1H','2S','2D','3H','6C']) self.assertEquals(['2','1'], c.isHouse()) def testTriple(self): c=Cards(['1S','1D','1H','2S','2D','3H','6C']) self.assertEquals('1', c.isTriple()) def testOnePair(self): c = Cards(['1S','1D','4H','2S','2D','3H','6C']) self.assertEquals('1S', c.isOnePair()) c = Cards(['1S','1D','4H','2S','5D','3H','6C']) self.assertEquals('1S', c.isOnePair()) def testTwoPair(self): c = Cards(['1S','1D','4H','2S','2D','3H','6C']) self.assertEquals(['2S','1S'], c.isTwoPair()) def testStraight(self): c = Cards(['1S','2D','3H','3S','4D','4H','5C']) self.assertEquals('5C', c.isStraight()) c = Cards(['1S','KD','QH','JS','0D','4H','5C']) self.assertEquals('1S', c.isStraight()) c = Cards(['8S','9D','JH','0S','7D','QH','5C']) self.assertEquals('QH', c.isStraight()) c = Cards(['2S','3D','4H','5S','7D','8H','5C']) self.assertEquals(None, c.isStraight()) c = Cards(['2S','3D','4H','5S','7D','7H','6S']) self.assertEquals('7D', c.isStraight()) def testStiFl(self): c = Cards(['2S','3S','4S','5S','7S','7H','6S']) self.assertEquals('7S', c.isStiFl()) c = Cards(['KS','0S','1S','JS','QS','7H','6S']) self.assertEquals('1S', c.isStiFl()) c = Cards(['KS','0S','1D','JS','QS','7H','6S']) self.assertEquals(None, c.isStiFl()) class CardsTotalJudgmentTest(unittest.TestCase): def testOnePair(self): c=Cards(['1S', '1C', '3D', '4S', '5S', '7C', '8C']) self.assertEquals(('OnePair','1S'), c.decide()) c=Cards(['2S', '1C', '3D', '4S', '8S', '7C', '7S']) self.assertEquals(('OnePair','7S'), c.decide()) def testTwoPair(self): c=Cards(['1S', '1C', '3D', '4S', '5S', '7C', '7S']) self.assertEquals(('TwoPair',['7S','1S']), c.decide()) c=Cards(['KD', 'KC', '3D', '4S', '5C', '5H', '8S']) self.assertEquals(('TwoPair',['5H','KD']), c.decide()) def testTriple(self): c=Cards(['1S', '1C', '1D', '4S', '5S', '7C', '2H']) self.assertEquals(('Triple','1'), c.decide()) def testStraight(self): c=Cards(['3S', '4C', '5D', '6S', '5S', '7C', '7H']) self.assertEquals(('Straight','7H'), c.decide()) def testFlush(self): c=Cards(['3S', '4S', '5S', '6S', '5S', '7C', '7H']) self.assertEquals(('Flush','6S'), c.decide()) def testPocker(self): c=Cards(['3S', '3C', '3H', '6S', '5S', '7C', '3D']) self.assertEquals(('Pocker','3S'), c.decide()) def testStiFl(self): c=Cards(['2C', '3C', '4C', '6C', '5C', '8C', '3D']) self.assertEquals(('StiFl','6C'), c.decide()) def testNone(self): c=Cards(['2C', '3C', '4C', '7S', '5C', '8D', 'KD']) self.assertEquals((None,'KD'), c.decide()) class JudgeWinner(unittest.TestCase): def testOneCard(self): input=['1S','2S','4S','KS'] self.assertEquals('1S', getHighestCard(input)) def testTwoCard(self): input=['4C','2S','4D','2C'] self.assertEquals('4D', getHighestCard(input)) def testOneJokbo(self): input=[('OnePair','1S'), ('Flush','6S'), (None, '1S')] self.assertEquals((1,'Flush'), judge(input)) def testTwoJokbo(self): input=[('OnePair','4C'), ('OnePair','4D'), (None, '2S')] self.assertEquals((1,'OnePair'), judge(input)) def testTwoTwoPair(self): input=[('TwoPair',['4C','1C']), ('OnePair','4D'), ('TwoPair',['2S','7C'])] self.assertEquals((0,'TwoPair'), judge(input)) def testTwoHouse(self): input=[('House',['4','K']), ('OnePair','4D'), ('House',['2','1'])] self.assertEquals((2,'House'), judge(input)) def testTwoNone(self): input=[(None,'4C'), (None,'4D'), (None,'2D')] self.assertEquals((1,None), judge(input)) class GameStatTest(unittest.TestCase): def testJudgeWinner(self): g=Game(4) result=g.judgeWinner() self.assertEquals(str, type(result[1])) def testGetResultRecord(self): g=Game(4) #record4=(winnerIndex,jokbo1,jokbo2,jokbo3,jokbo4,...) resultRecord=g.getResultRecord() self.assertEquals(int, type(resultRecord[0])) self.assertEquals(5, len(resultRecord)) def testGameStat(self): gs=GameStat(10,4) self.assertEquals(10,len(gs.getRawResult())) def emitProbEachJokbo(): gs=GameStat(20000,4) gs.calculateEmitProb() jokboFreqDict,n=gs.getEmitProb() for key,value in jokboFreqDict.items(): print key,'...',100*float(value)/n def winProbEachJokbo(aNumOfPlayers): gs=GameStat(20000,aNumOfPlayers) gs.calculateWinProb() jokboWinDict,jokboFreqDict=gs.getWinProb() for key,value in jokboFreqDict.items(): print key,'...',100*(float(jokboWinDict[key])/value) def testAll(): unittest.main(argv=('','-v')) def testPart(): myTests = [JudgeWinner('testOneJokbo'), JudgeWinner('testTwoJokbo'), ] suite = unittest.TestSuite() for eachTest in myTests: suite.addTest(eachTest) unittest.TextTestRunner().run(suite) if __name__=='__main__': testAll() #testPart() #emitProbEachJokbo() #winProbEachJokbo(3)