This package solves Agnes (Agnes Sorel) solitaire card games. It can be used to solve games having the rules implemented in the GNOME AisleRiot package and the rules attributed to Dalton in 1909 and Parlett in 1979 among others and to calculate win rates.
Project description
Summary
This package solves Agnes (Agnes Sorel) solitaire card games. It can be used to solve games having the rules implemented in the GNOME AisleRiot package and the rules attributed to Dalton in 1909 (with a minor variant in the table layout) and Parlett in 1979 among others [1–3] and to calculate win rates.
Example
import random
import quagnes
random.seed(12345)
n_reps = 1000
attributes = ['n_states_checked', 'n_deal', 'n_move_to_foundation',
'n_move_card_in_tableau','n_no_move_possible','max_score','max_depth',
'current_depth']
# header for the output file
print('rc,' + ','.join(attributes))
for rep in range(0, n_reps):
new_game=quagnes.Agnes()
rc = new_game.play()
# Write the return code and selected attributes from the game
out_str = (str(rc) + ',' +
','.join([ str(getattr(new_game, attr)) for attr in attributes ]))
print(out_str, flush=True)
# rc==1 are games that were won, but probably best to only print a
# limited number for debugging or for rule variants when
# win rate is low
#if rc==1:
# f = open(f'winners/win{rep}.txt', 'w', encoding='utf-8')
# fwrite(new_game.print_history())
# f.close()
Release Notes
Version 1.0.0
- Change name of
split_same_suit_runs
parameter tosplit_runs
and make parameter applicable regardless of the value ofmove_same_suit
. - Add optimization for use when the losing state set is disabled. Track for each pile the last move depth, the number of cards moved, whether the pile was moved from or to, and whether the bottom card in the moved pile could be played to the foundation at the time of the move. Do not allow moves that: (a) reverse a previous tableau move without an intervening move to foundation or deal; (b) make the last deal after a tableau move where neither pile is dealt to and neither file had a card played to foundation; (c) play the bottom card of the moved pile to the foundation after the tableau move if it could have been played at the time of the move (and there has been no subsequent move to the pile or deal onto the pile). These optimizations are only employed when the losing states set is disabled, as they use information not stored in the losing states set to determine whether a move can be played.
- Add optimization when moves to empty piles allow zero or multiple cards to reduce the space of moves to consider when the stock is empty or near empty (without changing the win possibility or maximum score that can be attained). In these rules, we use the term 'top card' to describe the highest-rank card in the pile being moved and 'same-color top card' to be the card with equal rank in the same-color suit. The rules are: (a) when moving runs by suit, never split a run between same-suit cards when the stock is empty; (b) never split a run between same-suit cards when neither pile can be covered by a future deal and the same-color top card is in the foundation; (c) when the stock is empty, force a move onto a card of the same suit where possible when we know the target pile cannot be needed to move the same-color top card (ie, the same-color top card is already in the foundation or in same-suit sequence or the card one rank higher than the same-color top card is already exposed at the bottom of a pile); (d) when the stock is empty, force a move to the foundation if the card being moved cannot be needed as a target to be moved onto (ie, if the card in the same-color suit that is one rank lower is already in the foundation or in same-suit sequence).
- Add new attributes created for these two optimizations in
_AgnesState
class (in_suit_seq
,last_in_pile
,_all_lmi
) and to output printed when state is printed. - Add
_AgnesState.hash_str
attribute to store the string representing the hash of the object that will be stored in the losing states and check loop set. Add_Agnes_state.update_hash_str()
function called after move is performed or undone to save the string inhash_str
. Previously, thehash_str
was recalculated multiple times. - Fix bug in
Agnes.print_history
as only half the history was being printed. - Use a single set to track whether loop has occured rather than a stack of
sets. Save result of check whether state is in the losing states set
so that we do not inefficiently (re-)insert the state into the losing
states set. Add attributes
is_loop
andis_loser
to_AgnesState
class to support this logic. - Created for C++ implementation. Move detailed information on the background, game rules, program methodology, and analysis of win rates to that repository from this README. Add win rates for rule variants where the empty rule is AnyRun or HighRun.
External Documentation
A C++ implementation written by this package author also exists in a git repository. Detailed information on the background, game rules, program methodology, and analysis of win rates are available there.
Using the Package
Games are simulated by creating an Agnes
object with the appropriate
parameters, running Agnes.play()
, and then using the return code and
extracting desired statistics from Agnes
class attributes.
Input Parameters
The following are the parameters used to construct Agnes
objects:
Parameter | Type | Description |
---|---|---|
deck_filename |
str |
Read deck from text file. If empty, a random deck will be used by calling random.shuffle and reversing the results. The text file should consist of 52 lines with each line formatted as (rank, suit) , where rank is in {0, 1, ..., 12} and suit is in {0, 1, 2, 3}. Suits are the same color if their value is equal modulus 2. The first card dealt is the base card. |
move_to_empty_pile |
str |
Rule for what can be moved to empty tableau piles. See following table for valid values. |
move_same_suit |
boolean |
If True, move runs by suit. If False (default), move runs by color. |
split_runs |
boolean |
If True, movable runs must be moved in their entirety. If False (default), a movable run can be split for a move). |
face_up |
boolean |
If True, deal all cards face-up in the tableau. If False (default), only the last card in each column. |
track_threshold |
int |
If the number of cards left in the stock is greater than or equal to this value (default = 0), track losing states in a single set for the whole game. This set can consume a lot of memory if some of the other options are chosen that allow a large number of moves (eg, move_to_empty_pile != 'none' ). |
print_states |
boolean |
Print detailed information about each state as moves are performed and undone. |
maximize_score |
boolean |
Maximize score. Disable optimization that inspects the layout of highest-ranked cards in the tableau to determine if game is unwinnable. |
max_states |
int |
Terminate game with return code 3 when number of states examined exceeds this threshold. 0 (default) means no threshold is used. |
Rule for Moving to Empty Piles
Value of move_to_empty_pile |
Description |
---|---|
'none' |
Empty piles can only be filled when dealing from stock (default). |
'any run' |
Empty piles can be filled by any movable card or movable run. |
'high run' |
Empty piles can be filled by any movable highest-rank card or movable run starting with a highest-rank card. |
'any 1' |
Empty piles can be filled by any movable single card, but not by a movable run of more than one card. |
'high 1' |
Empty piles can be filled by a single movable highest-rank card, but not by a movable run of more than one card. |
Output
Agnes.play()
Return Code and Agnes
Attributes
The return code of the Agnes.play()
function indicates whether the game is
winnable (return code = 1), not winnable (return code = 2), or the
simulation was terminated due to reaching the maximum number of states
specified by the max_states
parameter (return code = 3). The Agnes
object has attributes with the same name and type as the input parameters
that store the values of the parameters used during object construction.
In addition, the following attributes of Agnes
are updated when
Agnes.play()
is executed:
Attribute | Description |
---|---|
n_states_checked |
Count number of states checked (including repeats). |
n_deal |
Count number of deals performed (including those undone). |
n_foundation |
Count number of moves to foundation performed (including those undone). |
n_move_card_in_tableau |
Count number of moves in tableau performed (including those undone). |
max_score |
Maximum score found during play. (If moving to empty piles is not allowed, this will not be the true maximum, unless -z was specified to disable the optimization that tries to detect unwinnable games from the location of the highest-rank cards in the tableau.) |
max_depth |
Maximum depth of the search tree of moves. |
current_depth |
Depth of the search tree when the game was terminated. This value is always 0 for losing games. |
Detailed Program Output
When print_states=True
for the Agnes
object, Agnes.play()
will print
to standard output each state as moves are performed or undone.
Move: Move 2 card(s) from pile 1 to pile 3
piles:16#9-8-7##49-19-44##28-51-15-31##3-33-14-5-30-29##45-23-35##43-34-42-0-13-48-47-50##20-21-41-2-37#
depth:11, n_stock_left:16, valid_moves:[DealMove(), TablMove(from_=3, n_cards=2, to_=2, expose=False, tabltype=2)]
Foundations:[-1, -1, 1, 1]
T0:[] | [(9, 0), (8, 0), (7, 0)]
T1:[] | [(10, 3), (6, 1), (5, 3)]
T2:[] | [(2, 2), (12, 3), (2, 1), (5, 2)]
T3:[] | [(3, 0), (7, 2), (1, 1), (5, 0), (4, 2), (3, 2)]
T4:[] | [(6, 3), (10, 1), (9, 2)]
T5:[] | [(4, 3), (8, 2), (3, 3), (0, 0), (0, 1), (9, 3), (8, 3), (11, 3)]
T6:[] | [(7, 1), (8, 1), (2, 3), (2, 0), (11, 2)]
in_suit_seq:0011 0011 0000 0010 0010 0000 0000 1000 1001 0000 0000 0000 0000
last_in_pile:0011 0011 0000 0010 0000 0011 0000 1000 0000 0010 0000 0011 0000
Last move info:[10-2-T-T, 11-2-F-T, 0-0-F-T, 11-2-T-T, 10-2-F-T, 0-0-F-T, 9-1-F-T]
A line-by-line explanation of this output is provided in the git repository of the C++ implementation. However, note that this Python package does not write any messages to standard error.
References
[1] Agnes (card game). Wikipedia. https://en.wikipedia.org/wiki/Agnes_(card_game). Retrieved March 15, 2024.
[2] Dalton W (1909). "My favourite Patiences" in The Strand Magazine, Vol 38.
[3] Parlett D (1979). The Penguin Book of Patience. London: Penguin.
Disclosures
We are not affiliated with any of the books, websites, or applications discussed in this documentation, except of course for this Python package and previously-mentioned C++ implementation which we wrote.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file quagnes-1.0.0.tar.gz
.
File metadata
- Download URL: quagnes-1.0.0.tar.gz
- Upload date:
- Size: 65.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.0.0 CPython/3.11.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | ab071f95313b2c7426e27f285dfff4148a9508764813ce7b90a9ffc743ff581d |
|
MD5 | df8ff1f78b0689f787703874d4d3a143 |
|
BLAKE2b-256 | 98561342f6ed00fe7836423c078479bedbdf0d2a0e0a33d500be32ea5accca56 |
File details
Details for the file quagnes-1.0.0-py3-none-any.whl
.
File metadata
- Download URL: quagnes-1.0.0-py3-none-any.whl
- Upload date:
- Size: 51.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.0.0 CPython/3.11.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 779cde740ddbde1353ffc9e5927a5a67327580e05854c0d69b48c062c493b377 |
|
MD5 | ec8bb489621bb7977629cfff48d09c7b |
|
BLAKE2b-256 | a3399182f0ba6e265345e1e2e7c669f8645098619cc83a0a3b5cc66abb4dbe29 |