## vim:ts=4:et:nowrap ## ##---------------------------------------------------------------------------## ## ## PySol -- a Python Solitaire game ## ## Copyright (C) 2003 Markus Franz Xaver Johannes Oberhumer ## Copyright (C) 2002 Markus Franz Xaver Johannes Oberhumer ## Copyright (C) 2001 Markus Franz Xaver Johannes Oberhumer ## Copyright (C) 2000 Markus Franz Xaver Johannes Oberhumer ## Copyright (C) 1999 Markus Franz Xaver Johannes Oberhumer ## Copyright (C) 1998 Markus Franz Xaver Johannes Oberhumer ## All Rights Reserved. ## ## 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; see the file COPYING. ## If not, write to the Free Software Foundation, Inc., ## 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ## ## Markus F.X.J. Oberhumer ## ## http://www.oberhumer.com/pysol ## ##---------------------------------------------------------------------------## # imports import os, sys import time # PySol imports from settings import USE_FREECELL_SOLVER, FCS_COMMAND from mfxutil import destruct from util import KING # /*********************************************************************** # // HintInterface is an abstract class that defines the public # // interface - it only consists of the constructor # // and the getHints() method. # // # // The whole hint system is exclusively used by Game.getHints(). # ************************************************************************/ class HintInterface: # level == 0: show hint (key `H') # level == 1: show hint and display score value (key `Ctrl-H') # level == 2: demo def __init__(self, game, level): pass # Compute all hints for the current position. # Subclass responsibility. # # Returns a list of "atomic hints" - an atomic hint is a 7-tuple # (score, pos, ncards, from_stack, to_stack, text_color, forced_move). # # if ncards == 0: deal cards # elif from_stack == to_stack: flip card # else: move cards from from_stack to to_stack # # score, pos and text_color are only for debugging. # A forced_move is the next move that must be taken after this move # in order to avoid endless loops during demo play. # # Deal and flip may only happen if self.level >= 2 (i.e. demo). # # See Game.showHint() for more information. def getHints(self, taken_hint=None): return [] # /*********************************************************************** # // AbstractHint provides a useful framework for derived hint classes. # // # // Subclasses should override computeHints() # ************************************************************************/ class AbstractHint(HintInterface): def __init__(self, game, level): self.game = game self.level = level self.score_flatten_value = 0 if self.level == 0: self.score_flatten_value = 10000 # temporaries within getHints() self.bonus_color = None # self.__clones = [] self.reset() def __del__(self): self.reset() def reset(self): self.hints = [] self.max_score = 0 self.__destructClones() # # stack cloning # # Create a shallow copy of a stack. class AClonedStack: def __init__(self, stack, stackcards): # copy class identity self.__class__ = stack.__class__ # copy model data (reference copy) stack.copyModel(self) # set new cards (shallow copy of the card list) self.cards = stackcards[:] def ClonedStack(self, stack, stackcards): s = self.AClonedStack(stack, stackcards) self.__clones.append(s) return s def __destructClones(self): for s in self.__clones: s.__class__ = self.AClonedStack # restore orignal class destruct(s) self.__clones = [] # When computing hints for level 0, the scores are flattened # (rounded down) to a multiple of score_flatten_value. # # The idea is that hints will appear equal within a certain score range # so that the player will not get confused by the demo-intelligence. # # Pressing `Ctrl-H' (level 1) will preserve the score. def addHint(self, score, ncards, from_stack, to_stack, text_color=None, forced_move=None): if score < 0: return self.max_score = max(self.max_score, score) # add an atomic hint if self.score_flatten_value > 0: score = (score / self.score_flatten_value) * self.score_flatten_value if text_color is None: text_color = self.BLACK assert forced_move is None or len(forced_move) == 7 # pos is used for preserving the original sort order on equal scores pos = -len(self.hints) ah = (int(score), pos, ncards, from_stack, to_stack, text_color, forced_move) self.hints.append(ah) # clean up and return hints sorted by score def __returnHints(self): hints = self.hints self.reset() hints.sort() hints.reverse() return hints # # getHints() default implementation: # - handle forced moves # - try to flip face-down cards # - call computeHints() to do something useful # - try to deal cards # - clean up and return hints sorted by score # # Default scores for flip and deal moves. SCORE_FLIP = 100000 # 0..100000 SCORE_DEAL = 0 # 0..100000 def getHints(self, taken_hint=None): # 0) setup self.reset() game = self.game # 1) forced moves of the prev. taken hint have absolute priority if taken_hint and taken_hint[6]: return [taken_hint[6]] # 2) try if we can flip a card if self.level >= 2: for r in game.allstacks: if r.canFlipCard(): self.addHint(self.SCORE_FLIP, 1, r, r) if self.SCORE_FLIP >= 90000: return self.__returnHints() # 3) ask subclass to do something useful self.computeHints() # 4) try if we can deal cards if self.level >= 2: if game.canDealCards(): self.addHint(self.SCORE_DEAL, 0, game.s.talon, None) return self.__returnHints() # subclass def computeHints(self): pass # # utility shallMovePile() # # we move the pile if it is accepted by the target stack def _defaultShallMovePile(self, from_stack, to_stack, pile, rpile): if from_stack is to_stack or not to_stack.acceptsCards(from_stack, pile): return 0 return 1 # same, but check for loops def _cautiousShallMovePile(self, from_stack, to_stack, pile, rpile): if from_stack is to_stack or not to_stack.acceptsCards(from_stack, pile): return 0 # if len(rpile) == 0: return 1 # now check for loops rr = self.ClonedStack(from_stack, stackcards=rpile) if rr.acceptsCards(to_stack, pile): # the pile we are going to move could be moved back - # this is dangerous as we can create endless loops... return 0 return 1 # same, but only check for loops only when in demo mode def _cautiousDemoShallMovePile(self, from_stack, to_stack, pile, rpile): if from_stack is to_stack or not to_stack.acceptsCards(from_stack, pile): return 0 if self.level >= 2: # if len(rpile) == 0: return 1 # now check for loops rr = self.ClonedStack(from_stack, stackcards=rpile) if rr.acceptsCards(to_stack, pile): # the pile we are going to move could be moved back - # this is dangerous as we can create endless loops... return 0 return 1 shallMovePile = _defaultShallMovePile # # other utility methods # def _canDropAllCards(self, from_stack, stacks, stackcards): assert from_stack not in stacks return 0 # FIXME: this does not account for cards which are dropped herein cards = pile[:] cards.reverse() for card in cards: for s in stacks: if s is not from_stack: if s.acceptsCards(from_stack, [card]): break else: return 0 return 1 # # misc. constants # # score value so that the scores look nicer K = KING + 1 # text_color that will display the score (for debug with level 1) BLACK = "black" RED = "red" BLUE = "blue" # /*********************************************************************** # // # ************************************************************************/ class DefaultHint(AbstractHint): # The DefaultHint is optimized for Klondike type games # and also deals quite ok with other simple variants. # # But it completely lacks any specific strategy about game # types like Forty Thieves, FreeCell, Golf, Spider, ... # # BTW, we do not cheat ! # # bonus scoring used in _getXxxScore() below - subclass overrideable # def _preferHighRankMoves(self): return 0 # Basic bonus for moving a card. # Bonus must be in range 0..999 BONUS_DROP_CARD = 300 # 0..400 BONUS_SAME_SUIT_MOVE = 200 # 0..400 BONUS_NORMAL_MOVE = 100 # 0..400 def _getMoveCardBonus(self, r, t, pile, rpile): assert pile bonus = 0 if rpile: rr = self.ClonedStack(r, stackcards=rpile) if (rr.canDropCards(self.game.s.foundations))[0]: # the card below the pile can be dropped bonus = self.BONUS_DROP_CARD if t.cards and t.cards[-1].suit == pile[0].suit: # simple heuristics - prefer moving high-rank cards bonus = bonus + self.BONUS_SAME_SUIT_MOVE + (1 + pile[0].rank) elif self._preferHighRankMoves(): # simple heuristics - prefer moving high-rank cards bonus = bonus + self.BONUS_NORMAL_MOVE + (1 + pile[0].rank) elif rpile: # simple heuristics - prefer low-rank cards in rpile bonus = bonus + self.BONUS_NORMAL_MOVE + (self.K - rpile[-1].rank) else: # simple heuristics - prefer moving high-rank cards bonus = bonus + self.BONUS_NORMAL_MOVE + (1 + pile[0].rank) return bonus # Special bonus for facing up a card after the current move. # Bonus must be in range 0..9000 BONUS_FLIP_CARD = 1500 # 0..9000 def _getFlipSpecialBonus(self, r, t, pile, rpile): assert pile and rpile # The card below the pile can be flipped # (do not cheat and look at it !) # default: prefer a short rpile bonus = max(self.BONUS_FLIP_CARD - len(rpile), 0) return bonus # Special bonus for moving a pile from stack r to stack t. # Bonus must be in range 0..9000 BONUS_CREATE_EMPTY_ROW = 9000 # 0..9000 BONUS_CAN_DROP_ALL_CARDS = 4000 # 0..4000 BONUS_CAN_CREATE_EMPTY_ROW = 2000 # 0..4000 def _getMoveSpecialBonus(self, r, t, pile, rpile): # check if we will create an empty row if not rpile: return self.BONUS_CREATE_EMPTY_ROW # check if the card below the pile can be flipped if not rpile[-1].face_up: return self._getFlipSpecialBonus(r, t, pile, rpile) # check if all the cards below our pile could be dropped if self._canDropAllCards(r, self.game.s.foundations, stackcards=rpile): # we can drop the whole remaining pile # (and will create an empty row in the next move) ##print "BONUS_CAN_DROP_ALL_CARDS", r, pile, rpile self.bonus_color = self.RED return self.BONUS_CAN_DROP_ALL_CARDS + self.BONUS_CAN_CREATE_EMPTY_ROW # check if the cards below our pile are a whole row if r.canMoveCards(rpile): # could we move the remaining pile ? for x in self.game.s.rows: # note: we allow x == r here, because the pile # (currently at the top of r) will be # available in the next move if x is t or not x.cards: continue if x.acceptsCards(r, rpile): # we can create an empty row in the next move ##print "BONUS_CAN_CREATE_EMPTY_ROW", r, x, pile, rpile self.bonus_color = self.BLUE return self.BONUS_CAN_CREATE_EMPTY_ROW return 0 # # scoring used in getHints() - subclass overrideable # # Score for moving a pile from stack r to stack t. # Increased score should be in range 0..9999 def _getMovePileScore(self, score, color, r, t, pile, rpile): assert pile self.bonus_color = color b1 = self._getMoveSpecialBonus(r, t, pile, rpile) assert 0 <= b1 <= 9000 b2 = self._getMoveCardBonus(r, t, pile, rpile) assert 0 <= b2 <= 999 return score + b1 + b2, self.bonus_color # Score for moving a pile (usually a single card) from the WasteStack. def _getMoveWasteScore(self, score, color, r, t, pile, rpile): assert pile self.bonus_color = color score = 30000 if t.cards: score = 31000 b2 = self._getMoveCardBonus(r, t, pile, rpile) assert 0 <= b2 <= 999 return score + b2, self.bonus_color # Score for dropping ncards from stack r to stack t. def _getDropCardScore(self, score, color, r, t, ncards): assert t is not r if ncards > 1: # drop immediately (Spider) return 93000, color pile = r.cards c = pile[-1] # compute distance to t.cap.base_rank - compare Stack.getRankDir() if t.cap.base_rank < 0: d = len(t.cards) else: d = (c.rank - t.cap.base_rank) % t.cap.mod if d > t.cap.mod / 2: d = d - t.cap.mod if abs(d) <= 1: # drop Ace and 2 immediately score = 92000 elif r in self.game.sg.talonstacks: score = 25000 # less than _getMoveWasteScore() elif len(pile) == 1: ###score = 50000 score = 91000 elif self._canDropAllCards(r, self.game.s.foundations, stackcards=pile[:-1]): score = 90000 color = self.RED else: # don't drop this card too eagerly - we may need it # for pile moving score = 50000 score = score + (self.K - c.rank) return score, color # # compute hints - main hint intelligence # def computeHints(self): game = self.game # 1) check Tableau piles self.step010(game.sg.dropstacks, game.s.rows) # 2) try if we can move part of a pile within the RowStacks # so that we can drop a card afterwards if not self.hints and self.level >= 1: self.step020(game.s.rows, game.s.foundations) # 3) try if we should move a card from a Foundation to a RowStack if not self.hints and self.level >= 1: self.step030(game.s.foundations, game.s.rows, game.sg.dropstacks) # 4) try if we can move a card from a RowStack to a ReserveStack if not self.hints: self.step040(game.s.rows, game.sg.reservestacks) # 5) try if we should move a card from a ReserveStack to a RowStack if not self.hints: self.step050(game.sg.reservestacks, game.s.rows) # Don't be too clever and give up ;-) # # implementation of the hint steps # # 1) check Tableau piles def step010(self, dropstacks, rows): # for each stack for r in dropstacks: # 1a) try if we can drop cards t, ncards = r.canDropCards(self.game.s.foundations) if t: score, color = 0, None score, color = self._getDropCardScore(score, color, r, t, ncards) self.addHint(score, ncards, r, t, color) if score >= 90000: break # 1b) try if we can move cards to one of the RowStacks for pile in self.step010b_getPiles(r): if pile: self.step010_movePile(r, pile, rows) def step010b_getPiles(self, stack): # return all moveable piles for this stack, longest one first return (stack.getPile(), ) def step010_movePile(self, r, pile, rows): lp = len(pile) lr = len(r.cards) assert 1 <= lp <= lr rpile = r.cards[ : (lr-lp) ] # remaining pile empty_row_seen = 0 r_is_waste = r in self.game.sg.talonstacks for t in rows: score, color = 0, None if not self.shallMovePile(r, t, pile, rpile): continue if r_is_waste: # moving a card from the WasteStack score, color = self._getMoveWasteScore(score, color, r, t, pile, rpile) else: if not t.cards: # the target stack is empty if lp == lr: # do not move a whole stack from row to row continue if empty_row_seen: # only make one hint for moving to an empty stack # (in case we have multiple empty stacks) continue score = 60000 empty_row_seen = 1 else: # the target stack is not empty score = 80000 score, color = self._getMovePileScore(score, color, r, t, pile, rpile) self.addHint(score, lp, r, t, color) # 2) try if we can move part of a pile within the RowStacks # so that we can drop a card afterwards # score: 40000 .. 59999 step020_getPiles = step010b_getPiles def step020(self, rows, foundations): for r in rows: for pile in self.step020_getPiles(r): if not pile or len(pile) < 2: continue # is there a card in our pile that could be dropped ? drop_info = [] i = 0 for c in pile: rr = self.ClonedStack(r, stackcards=[c]) stack, ncards = rr.canDropCards(foundations) if stack and stack is not r: assert ncards == 1 drop_info.append((c, stack, ncards, i)) i = i + 1 # now try to make a move so that the drop-card will get free for di in drop_info: c = di[0] sub_pile = pile[di[3]+1 : ] ##print "trying drop move", c, pile, sub_pile ##assert r.canMoveCards(sub_pile) if not r.canMoveCards(sub_pile): continue for t in rows: if t is r or not t.acceptsCards(r, sub_pile): continue ##print "drop move", r, t, sub_pile score = 40000 score = score + 1000 + (self.K - r.getCard().rank) # force the drop (to avoid loops) force = (999999, 0, di[2], r, di[1], self.BLUE, None) self.addHint(score, len(sub_pile), r, t, self.RED, forced_move=force) # 3) try if we should move a card from a Foundation to a RowStack # score: 20000 .. 29999 def step030(self, foundations, rows, dropstacks): for s in foundations: card = s.getCard() if not card or not s.canMoveCards([card]): continue # search a RowStack that would accept the card for t in rows: if t is s or not t.acceptsCards(s, [card]): continue tt = self.ClonedStack(t, stackcards=t.cards+[card]) # search a Stack that would benefit from this card for r in dropstacks: if r is t: continue pile = r.getPile() if not pile: continue if not tt.acceptsCards(r, pile): continue # compute remaining pile in r rpile = r.cards[ : (len(r.cards)-len(pile)) ] rr = self.ClonedStack(r, stackcards=rpile) if rr.acceptsCards(t, pile): # the pile we are going to move from r to t # could be moved back from t ro r - this is # dangerous as we can create loops... continue score = 20000 + card.rank ##print score, s, t, r, pile, rpile # force the move from r to t (to avoid loops) force = (999999, 0, len(pile), r, t, self.BLUE, None) self.addHint(score, 1, s, t, self.BLUE, forced_move=force) # 4) try if we can move a card from a RowStack to a ReserveStack # score: 10000 .. 19999 def step040(self, rows, reservestacks): if not reservestacks: return for r in rows: card = r.getCard() if not card or not r.canMoveCards([card]): continue pile = [card] # compute remaining pile in r rpile = r.cards[ : (len(r.cards)-len(pile)) ] rr = self.ClonedStack(r, stackcards=rpile) for t in reservestacks: if t is r or not t.acceptsCards(r, pile): continue if rr.acceptsCards(t, pile): # the pile we are going to move from r to t # could be moved back from t ro r - this is # dangerous as we can create loops... continue score = 10000 score, color = self._getMovePileScore(score, None, r, t, pile, rpile) self.addHint(score, len(pile), r, t, color) break # 5) try if we should move a card from a ReserveStack to a RowStack def step050(self, reservestacks, rows): if not reservestacks: return ### FIXME # /*********************************************************************** # // # ************************************************************************/ class CautiousDefaultHint(DefaultHint): shallMovePile = DefaultHint._cautiousShallMovePile ##shallMovePile = DefaultHint._cautiousDemoShallMovePile def _preferHighRankMoves(self): return 1 # /*********************************************************************** # // now some default hints for the various game types # ************************************************************************/ # DefaultHint is optimized for Klondike type games anyway class KlondikeType_Hint(DefaultHint): pass # this works for Yukon, but not too well for Russian Solitaire class YukonType_Hint(CautiousDefaultHint): def step010b_getPiles(self, stack): # return all moveable piles for this stack, longest one first p = stack.getPile() piles = [] while p: piles.append(p) p = p[1:] # note: we need a fresh shallow copy return piles # FIXME class FreeCellType_Hint(CautiousDefaultHint): pass class GolfType_Hint(DefaultHint): pass class SpiderType_Hint(DefaultHint): pass # /*********************************************************************** # // FreeCell-Solver # ************************************************************************/ class FreeCellSolver_Hint: def __init__(self, game, dialog, **game_type): self.game = game self.dialog = dialog self.game_type = game_type self.options = { 'method': 'dfs', 'max_iters': 10000, 'max_depth': 1000, 'progress': False, 'preset': None, } self.hints = [] self.hints_index = 0 # correct cards rank if foundations.base_rank != 0 (Penguin, Opus) if 'base_rank' in game_type: # (Simple Simon) self.base_rank = game_type['base_rank'] else: self.base_rank = game.s.foundations[0].cap.base_rank ##print 'game_type:', game_type ##print 'base_rank:', self.base_rank def config(self, **kw): self.options.update(kw) def card2str1(self, card): # row and reserves rank = (card.rank-self.base_rank) % 13 return "A23456789TJQK"[rank] + "CSHD"[card.suit] def card2str2(self, card): # foundations rank = (card.rank-self.base_rank) % 13 return "CSHD"[card.suit] + "-" + "A23456789TJQK"[rank] # hard solvable: Freecell #47038300998351211829 (65539 iters) def getHints(self, taken_hint=None): if taken_hint and taken_hint[6]: return [taken_hint[6]] h = self.hints[self.hints_index] #print 'getHints', taken_hint, h if h is None: return None ncards, src, dest = h thint = None if len(src.cards) > ncards and not src.cards[-ncards-1].face_up: # flip card thint = (999999, 0, 1, src, src, None, None) if dest == None: # foundation cards = src.cards[-ncards:] for f in self.game.s.foundations: if f.acceptsCards(src, cards): dest = f break assert dest hint = (999999, 0, ncards, src, dest, None, thint) self.hints_index += 1 #print hint return [hint] def computeHints(self): game = self.game game_type = self.game_type progress = self.options['progress'] board = '' # # b = '' for s in self.game.s.foundations: if s.cards: ss = self.card2str2(s.cards[-1]) b += ' ' + ss if b: board += 'Founds:' + b + '\n' # b = '' for s in self.game.s.reserves: if s.cards: cs = self.card2str1(s.cards[-1]) b += ' ' + cs else: b += ' -' if b: board += 'FC:' + b + '\n' # for s in self.game.s.rows: b = '' for c in s.cards: cs = self.card2str1(c) if not c.face_up: cs = '<%s>' % cs b += cs + ' ' board = board + b.strip() + '\n' # ##print '--------------------\n', board, '--------------------' # args = [] ##args += ['-sam', '-p', '-opt', '--display-10-as-t'] args += ['-m', '-p', '-opt'] if self.options['preset'] and self.options['preset'] != 'none': args += ['-l', self.options['preset']] if progress: args += ['--iter-output'] ##args += ['-s'] args += ['--max-iters', self.options['max_iters'], '--max-depth', self.options['max_depth'], '--method', self.options['method'], '--decks-num', game.gameinfo.decks, '--stacks-num', len(game.s.rows), '--freecells-num', len(game.s.reserves), ] # if 'preset' in game_type: args += ['--preset', game_type['preset']] if 'sbb' in game_type: args += ['--sequences-are-built-by', game_type['sbb']] if 'sm' in game_type: args += ['--sequence-move', game_type['sm']] if 'esf' in game_type: args += ['--empty-stacks-filled-by', game_type['esf']] command = FCS_COMMAND+' '+' '.join([str(i) for i in args]) ##print command pin, pout, perr = os.popen3(command) pin.write(board) pin.close() # stack_types = { 'the' : game.s.foundations, 'stack' : game.s.rows, 'freecell' : game.s.reserves, } ##start_time = time.time() if progress: # iteration output iter = 0 depth = 0 states = 0 for s in pout: ##print s, if s.startswith('Iter'): #print `s` iter = int(s[10:-1]) elif s.startswith('Depth'): #print s, depth = int(s[6:-1]) elif s.startswith('Stor'): #print s, states = int(s[14:-1]) if iter % 100 == 0: self.dialog.setText(iter=iter, depth=depth, states=states) elif s.startswith('-=-') or \ s.startswith('I co'): # "I could not solve this game." break self.dialog.setText(iter=iter, depth=depth, states=states) hints = [] for s in pout: #print s, # TODO: # Total number of states checked is 6. # This scan generated 6 states. if not s.startswith('Move'): continue words = s.split() ncards = words[1] if ncards == 'the': # "Move the sequence on top of Stack 1 to the foundations" # (Simple Simon) ncards = 0 elif ncards == 'a': ncards = 1 else: ncards = int(ncards) if ncards: st = stack_types[words[4]] sn = int(words[5]) src = st[sn] # source stack if words[7] == 'the': # to foundation dest = None else: # to rows or reserves dt = stack_types[words[7]] dn = int(words[8]) dest = dt[dn] else: # move sequence ncards = 13 st = stack_types['stack'] sn = int(words[7]) src = st[sn] dest = None hints.append([ncards, src, dest]) ##print src, dest, ncards # ##print 'time:', time.time()-start_time ##print perr.read(), self.hints = hints self.hints.append(None) # XXX ##print self.hints pout.close() perr.close() if os.name == 'posix': os.wait() class FreeCellSolverWrapper: def __init__(self, **game_type): self.game_type = game_type def __call__(self, game, dialog): hint = FreeCellSolver_Hint(game, dialog, **self.game_type) return hint