Skip to content

Commit

Permalink
Refactor _evaluate_omaha_cards and simplify it (#110)
Browse files Browse the repository at this point in the history
* Refactor _evaluate_omaha_cards and simplify it

* Magic Numbers replaced by constants

* Reduntant constant values removed

---------

Co-authored-by: root <root@LAPTOP-264SBR7M>
  • Loading branch information
ParthShitole and root authored Oct 19, 2024
1 parent bb176e8 commit adbcca1
Showing 1 changed file with 63 additions and 48 deletions.
111 changes: 63 additions & 48 deletions python/phevaluator/evaluator_omaha.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,11 @@

COMMUNITY_CARD_COUNT = 5
HOLE_CARD_COUNT = 4
MAX_RANK_VALUE = 10000
MIN_FLUSH_COUNT_BOARD = 3
MIN_FLUSH_COUNT_HOLE = 2
MAX_FLUSH_HOLE_HASH = 1365
MAX_NO_FLUSH_HOLE_HASH = 1820
TOTAL_CARD_COUNT = COMMUNITY_CARD_COUNT + HOLE_CARD_COUNT


Expand Down Expand Up @@ -64,61 +69,73 @@ def evaluate_omaha_cards(*cards: int | str | Card) -> int:

# TODO(@azriel1rf): `_evaluate_omaha_cards` is too complex. Consider refactoring.
# https://github.com/HenryRLee/PokerHandEvaluator/issues/92
def _evaluate_omaha_cards(community_cards: list[int], hole_cards: list[int]) -> int: # noqa: C901, PLR0912
value_flush = 10000
value_noflush = 10000
suit_count_board = [0] * 4
suit_count_hole = [0] * 4
def _evaluate_omaha_cards(community_cards: list[int], hole_cards: list[int]) -> int:
value_flush = MAX_RANK_VALUE
suit_count_board = _count_suits(community_cards)
suit_count_hole = _count_suits(hole_cards)

for community_card in community_cards:
suit_count_board[community_card % 4] += 1
flush_suit = _determine_flush_suit(suit_count_board, suit_count_hole)

for hole_card in hole_cards:
suit_count_hole[hole_card % 4] += 1
if flush_suit != -1:
value_flush = _determine_flush_value(
community_cards, hole_cards, suit_count_board, suit_count_hole, flush_suit
)

min_flush_count_board = 3
min_flush_count_hole = 2
value_noflush = _determine_noflush_value(community_cards, hole_cards)

flush_suit = -1
for i in range(4):
if (
suit_count_board[i] >= min_flush_count_board
and suit_count_hole[i] >= min_flush_count_hole
):
flush_suit = i
break
return min(value_flush, value_noflush)

if flush_suit != -1:
flush_count_board = suit_count_board[flush_suit]
flush_count_hole = suit_count_hole[flush_suit]

suit_binary_board = 0
for community_card in community_cards:
if community_card % 4 == flush_suit:
suit_binary_board |= BINARIES_BY_ID[community_card]
def _count_suits(cards: list[int]) -> list[int]:
suit_count = [0] * 4
for card in cards:
suit_count[card % 4] += 1
return suit_count

suit_binary_hole = 0
for hole_card in hole_cards:
if hole_card % 4 == flush_suit:
suit_binary_hole |= BINARIES_BY_ID[hole_card]

def _determine_flush_suit(
suit_count_board: list[int], suit_count_hole: list[int]
) -> int:
for i in range(4):
if (
flush_count_board == min_flush_count_board
and flush_count_hole == min_flush_count_hole
suit_count_board[i] >= MIN_FLUSH_COUNT_BOARD
and suit_count_hole[i] >= MIN_FLUSH_COUNT_HOLE
):
value_flush = FLUSH[suit_binary_board | suit_binary_hole]

else:
padding = [0x0000, 0x2000, 0x6000]

suit_binary_board |= padding[COMMUNITY_CARD_COUNT - flush_count_board]
suit_binary_hole |= padding[HOLE_CARD_COUNT - flush_count_hole]

board_hash = hash_binary(suit_binary_board, COMMUNITY_CARD_COUNT)
hole_hash = hash_binary(suit_binary_hole, HOLE_CARD_COUNT)

value_flush = FLUSH_OMAHA[board_hash * 1365 + hole_hash]

return i
return -1


def _determine_flush_value(
community_cards: list[int],
hole_cards: list[int],
suit_count_board: list[int],
suit_count_hole: list[int],
flush_suit: int,
) -> int:
flush_count_board = suit_count_board[flush_suit]
flush_count_hole = suit_count_hole[flush_suit]
suit_binary_board = 0
for community_card in community_cards:
if community_card % 4 == flush_suit:
suit_binary_board |= BINARIES_BY_ID[community_card]
suit_binary_hole = 0
for hole_card in hole_cards:
if hole_card % 4 == flush_suit:
suit_binary_hole |= BINARIES_BY_ID[hole_card]
if (
flush_count_board == MIN_FLUSH_COUNT_BOARD
and flush_count_hole == MIN_FLUSH_COUNT_HOLE
):
return FLUSH[suit_binary_board | suit_binary_hole]
padding = [0x0000, 0x2000, 0x6000]
suit_binary_board |= padding[COMMUNITY_CARD_COUNT - flush_count_board]
suit_binary_hole |= padding[HOLE_CARD_COUNT - flush_count_hole]
board_hash = hash_binary(suit_binary_board, COMMUNITY_CARD_COUNT)
hole_hash = hash_binary(suit_binary_hole, HOLE_CARD_COUNT)
return FLUSH_OMAHA[board_hash * MAX_FLUSH_HOLE_HASH + hole_hash]


def _determine_noflush_value(community_cards: list[int], hole_cards: list[int]) -> int:
quinary_board = [0] * 13
quinary_hole = [0] * 13

Expand All @@ -131,6 +148,4 @@ def _evaluate_omaha_cards(community_cards: list[int], hole_cards: list[int]) ->
board_hash = hash_quinary(quinary_board, COMMUNITY_CARD_COUNT)
hole_hash = hash_quinary(quinary_hole, HOLE_CARD_COUNT)

value_noflush = NO_FLUSH_OMAHA[board_hash * 1820 + hole_hash]

return min(value_flush, value_noflush)
return NO_FLUSH_OMAHA[board_hash * MAX_NO_FLUSH_HOLE_HASH + hole_hash]

0 comments on commit adbcca1

Please sign in to comment.