python-zulip-api/zulip_bots/zulip_bots/bots/chessbot/chessbot.py
Anders Kaseorg 189cf48573 chessbot: Upgrade python-chess to chess.
Signed-off-by: Anders Kaseorg <anders@zulip.com>
2021-08-24 19:01:58 -07:00

638 lines
22 KiB
Python

import copy
import re
from typing import Dict, Optional
import chess
import chess.engine
from zulip_bots.lib import BotHandler
START_REGEX = re.compile("start with other user$")
START_COMPUTER_REGEX = re.compile("start as (?P<user_color>white|black) with computer")
MOVE_REGEX = re.compile("do (?P<move_san>.+)$")
RESIGN_REGEX = re.compile("resign$")
class ChessHandler:
def usage(self) -> str:
return (
"Chess Bot is a bot that allows you to play chess against either "
"another user or the computer. Use `start with other user` or "
"`start as <color> with computer` to start a game.\n\n"
"In order to play against a computer, `chess.conf` must be set "
"with the key `stockfish_location` set to the location of the "
"Stockfish program on this computer."
)
def initialize(self, bot_handler: BotHandler) -> None:
self.config_info = bot_handler.get_config_info("chess")
try:
self.engine = chess.engine.SimpleEngine.popen_uci(
self.config_info["stockfish_location"]
)
except FileNotFoundError:
# It is helpful to allow for fake Stockfish locations if the bot
# runner is testing or knows they won't be using an engine.
print("That Stockfish doesn't exist. Continuing.")
def handle_message(self, message: Dict[str, str], bot_handler: BotHandler) -> None:
content = message["content"]
if content == "":
bot_handler.send_reply(message, self.usage())
return
start_regex_match = START_REGEX.match(content)
start_computer_regex_match = START_COMPUTER_REGEX.match(content)
move_regex_match = MOVE_REGEX.match(content)
resign_regex_match = RESIGN_REGEX.match(content)
is_with_computer = False
last_fen = chess.Board().fen()
if bot_handler.storage.contains("is_with_computer"):
is_with_computer = (
# `bot_handler`'s `storage` only accepts `str` values.
bot_handler.storage.get("is_with_computer")
== str(True)
)
if bot_handler.storage.contains("last_fen"):
last_fen = bot_handler.storage.get("last_fen")
if start_regex_match:
self.start(message, bot_handler)
elif start_computer_regex_match:
self.start_computer(
message, bot_handler, start_computer_regex_match.group("user_color") == "white"
)
elif move_regex_match:
if is_with_computer:
self.move_computer(
message, bot_handler, last_fen, move_regex_match.group("move_san")
)
else:
self.move(message, bot_handler, last_fen, move_regex_match.group("move_san"))
elif resign_regex_match:
self.resign(message, bot_handler, last_fen)
def start(self, message: Dict[str, str], bot_handler: BotHandler) -> None:
"""Starts a game with another user, with the current user as white.
Replies to the bot handler.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
"""
new_board = chess.Board()
bot_handler.send_reply(message, make_start_reponse(new_board))
# `bot_handler`'s `storage` only accepts `str` values.
bot_handler.storage.put("is_with_computer", str(False))
bot_handler.storage.put("last_fen", new_board.fen())
def start_computer(
self, message: Dict[str, str], bot_handler: BotHandler, is_white_user: bool
) -> None:
"""Starts a game with the computer. Replies to the bot handler.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- is_white_user: Whether or not the player wants to be
white. If false, the user is black. If the
user is white, they will get to make the
first move; if they are black the computer
will make the first move.
"""
new_board = chess.Board()
if is_white_user:
bot_handler.send_reply(message, make_start_computer_reponse(new_board))
# `bot_handler`'s `storage` only accepts `str` values.
bot_handler.storage.put("is_with_computer", str(True))
bot_handler.storage.put("last_fen", new_board.fen())
else:
self.move_computer_first(
message,
bot_handler,
new_board.fen(),
)
def validate_board(
self, message: Dict[str, str], bot_handler: BotHandler, fen: str
) -> Optional[chess.Board]:
"""Validates a board based on its FEN string. Replies to the bot
handler if there is an error with the board.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- fen: The FEN string of the board.
Returns: `None` if the board didn't pass, or the board object itself
if it did.
"""
try:
last_board = chess.Board(fen)
except ValueError:
bot_handler.send_reply(message, make_copied_wrong_response())
return None
return last_board
def validate_move(
self,
message: Dict[str, str],
bot_handler: BotHandler,
last_board: chess.Board,
move_san: str,
is_computer: object,
) -> Optional[chess.Move]:
"""Validates a move based on its SAN string and the current board.
Replies to the bot handler if there is an error with the move.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- last_board: The board object before the move.
- move_san: The SAN of the move.
- is_computer: Whether or not the user is playing against a
computer (used in the response if the move is not
legal).
Returns: `False` if the move didn't pass, or the move object itself if
it did.
"""
try:
move = last_board.parse_san(move_san)
except ValueError:
bot_handler.send_reply(message, make_not_legal_response(last_board, move_san))
return None
if move not in last_board.legal_moves:
bot_handler.send_reply(message, make_not_legal_response(last_board, move_san))
return None
return move
def check_game_over(
self, message: Dict[str, str], bot_handler: BotHandler, new_board: chess.Board
) -> bool:
"""Checks if a game is over due to
- checkmate,
- stalemate,
- insufficient material,
- 50 moves without a capture or pawn move, or
- 3-fold repetition.
Replies to the bot handler if it is game over.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- new_board: The board object.
Returns: True if it is game over, false if it's not.
"""
# This assumes that the players will claim a draw after 3-fold
# repetition or 50 moves go by without a capture or pawn move.
# According to the official rules, the game is only guaranteed to
# be over if it's *5*-fold or *75* moves, but if either player
# wants the game to be a draw, after 3 or 75 it a draw. For now,
# just assume that the players would want the draw.
if new_board.is_game_over(claim_draw=True):
game_over_output = ""
if new_board.is_checkmate():
game_over_output = make_loss_response(new_board, "was checkmated")
elif new_board.is_stalemate():
game_over_output = make_draw_response("stalemate")
elif new_board.is_insufficient_material():
game_over_output = make_draw_response("insufficient material")
elif new_board.can_claim_fifty_moves():
game_over_output = make_draw_response("50 moves without a capture or pawn move")
elif new_board.can_claim_threefold_repetition():
game_over_output = make_draw_response("3-fold repetition")
bot_handler.send_reply(message, game_over_output)
return True
return False
def move(
self, message: Dict[str, str], bot_handler: BotHandler, last_fen: str, move_san: str
) -> None:
"""Makes a move for a user in a game with another user. Replies to
the bot handler.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- last_fen: The FEN string of the board before the move.
- move_san: The SAN of the move to make.
"""
last_board = self.validate_board(message, bot_handler, last_fen)
if not last_board:
return
move = self.validate_move(message, bot_handler, last_board, move_san, False)
if not move:
return
new_board = copy.copy(last_board)
new_board.push(move)
if self.check_game_over(message, bot_handler, new_board):
return
bot_handler.send_reply(message, make_move_reponse(last_board, new_board, move))
bot_handler.storage.put("last_fen", new_board.fen())
def move_computer(
self, message: Dict[str, str], bot_handler: BotHandler, last_fen: str, move_san: str
) -> None:
"""Preforms a move for a user in a game with the computer and then
makes the computer's move. Replies to the bot handler. Unlike `move`,
replies only once to the bot handler every two moves (only after the
computer moves) instead of after every move. Doesn't require a call in
order to make the computer move. To make the computer move without the
user going first, use `move_computer_first`.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- last_fen: The FEN string of the board before the user's move.
- move_san: The SAN of the user's move to make.
"""
last_board = self.validate_board(message, bot_handler, last_fen)
if not last_board:
return
move = self.validate_move(message, bot_handler, last_board, move_san, True)
if not move:
return
new_board = copy.copy(last_board)
new_board.push(move)
if self.check_game_over(message, bot_handler, new_board):
return
computer_move = calculate_computer_move(new_board, self.engine)
if not computer_move:
bot_handler.send_reply(message, make_engine_failed_response())
return
new_board_after_computer_move = copy.copy(new_board)
new_board_after_computer_move.push(computer_move)
if self.check_game_over(message, bot_handler, new_board_after_computer_move):
return
bot_handler.send_reply(
message, make_move_reponse(new_board, new_board_after_computer_move, computer_move)
)
bot_handler.storage.put("last_fen", new_board_after_computer_move.fen())
def move_computer_first(
self, message: Dict[str, str], bot_handler: BotHandler, last_fen: str
) -> None:
"""Preforms a move for the computer without having the user go first in
a game with the computer. Replies to the bot handler. Like
`move_computer`, but doesn't have the user move first. This is usually
only useful at the beginning of a game.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- last_fen: The FEN string of the board before the computer's
move.
"""
last_board = self.validate_board(message, bot_handler, last_fen)
if not last_board:
return
computer_move = calculate_computer_move(last_board, self.engine)
if not computer_move:
bot_handler.send_reply(message, make_engine_failed_response())
return
new_board_after_computer_move = copy.copy(last_board)
new_board_after_computer_move.push(computer_move)
if self.check_game_over(message, bot_handler, new_board_after_computer_move):
return
bot_handler.send_reply(
message, make_move_reponse(last_board, new_board_after_computer_move, computer_move)
)
bot_handler.storage.put("last_fen", new_board_after_computer_move.fen())
# `bot_handler`'s `storage` only accepts `str` values.
bot_handler.storage.put("is_with_computer", str(True))
def resign(self, message: Dict[str, str], bot_handler: BotHandler, last_fen: str) -> None:
"""Resigns the game for the current player.
Parameters:
- message: The Zulip Bots message object.
- bot_handler: The Zulip Bots bot handler object.
- last_fen: The FEN string of the board.
"""
last_board = self.validate_board(message, bot_handler, last_fen)
if not last_board:
return
bot_handler.send_reply(message, make_loss_response(last_board, "resigned"))
handler_class = ChessHandler
def calculate_computer_move(
board: chess.Board, engine: chess.engine.SimpleEngine
) -> Optional[chess.Move]:
"""Calculates the computer's move.
Parameters:
- board: The board object before the move.
- engine: The UCI engine object.
Returns: The computer's move object.
"""
result = engine.play(board, chess.engine.Limit(time=3.0))
return result.move
def make_draw_response(reason: str) -> str:
"""Makes a response string for a draw.
Parameters:
- reason: The reason for the draw, in the form of a noun, e.g.,
'stalemate' or 'insufficient material'.
Returns: The draw response string.
"""
return f"It's a draw because of {reason}!"
def make_loss_response(board: chess.Board, reason: str) -> str:
"""Makes a response string for a loss (or win).
Parameters:
- board: The board object at the end of the game.
- reason: The reason for the loss, in the form of a predicate, e.g.,
'was checkmated'.
Returns: The loss response string.
"""
return ("*{}* {}. **{}** wins!\n\n" "{}").format(
"White" if board.turn else "Black",
reason,
"Black" if board.turn else "White",
make_str(board, board.turn),
)
def make_not_legal_response(board: chess.Board, move_san: str) -> str:
"""Makes a response string for a not-legal move.
Parameters:
- board: The board object before the move.
- move_san: The SAN of the not-legal move.
Returns: The not-legal-move response string.
"""
return ("Sorry, the move *{}* isn't legal.\n\n" "{}" "\n\n\n" "{}").format(
move_san, make_str(board, board.turn), make_footer()
)
def make_copied_wrong_response() -> str:
"""Makes a response string for a FEN string that was copied wrong.
Returns: The copied-wrong response string.
"""
return (
"Sorry, it seems like you copied down the response wrong.\n\n"
"Please try to copy the response again from the last message!"
)
def make_start_reponse(board: chess.Board) -> str:
"""Makes a response string for the first response of a game with another
user.
Parameters:
- board: The board object to start the game with (which most-likely
should be general opening chess position).
Returns: The starting response string.
"""
return (
"New game! The board looks like this:\n\n"
"{}"
"\n\n\n"
"Now it's **{}**'s turn."
"\n\n\n"
"{}"
).format(make_str(board, True), "white" if board.turn else "black", make_footer())
def make_start_computer_reponse(board: chess.Board) -> str:
"""Makes a response string for the first response of a game with a
computer, when the user is playing as white. If the user is playing as
black, use `ChessHandler.move_computer_first`.
Parameters:
- board: The board object to start the game with (which most-likely
should be general opening chess position).
Returns: The starting response string.
"""
return (
"New game with computer! The board looks like this:\n\n"
"{}"
"\n\n\n"
"Now it's **{}**'s turn."
"\n\n\n"
"{}"
).format(make_str(board, True), "white" if board.turn else "black", make_footer())
def make_move_reponse(last_board: chess.Board, new_board: chess.Board, move: chess.Move) -> str:
"""Makes a response string for after a move is made.
Parameters:
- last_board: The board object before the move.
- new_board: The board object after the move.
- move: The move object.
Returns: The move response string.
"""
return (
"The board was like this:\n\n"
"{}"
"\n\n\n"
"Then *{}* moved *{}*:\n\n"
"{}"
"\n\n\n"
"Now it's **{}**'s turn."
"\n\n\n"
"{}"
).format(
make_str(last_board, new_board.turn),
"white" if last_board.turn else "black",
last_board.san(move),
make_str(new_board, new_board.turn),
"white" if new_board.turn else "black",
make_footer(),
)
def make_engine_failed_response() -> str:
"""Makes a response string for engine failure.
Returns: The engine failure response string.
"""
return "The computer failed to make a move."
def make_footer() -> str:
"""Makes a footer to be appended to the bottom of other, actionable
responses.
"""
return (
"To make your next move, respond to Chess Bot with\n\n"
"```do <your move>```\n\n"
"*Remember to @-mention Chess Bot at the beginning of your "
"response.*"
)
def make_str(board: chess.Board, is_white_on_bottom: bool) -> str:
"""Converts a board object into a string to be used in Markdown. Backticks
are added around the string to preserve formatting.
Parameters:
- board: The board object.
- is_white_on_bottom: Whether or not white should be on the bottom
side in the string. If false, black will be on
the bottom.
Returns: The string made from the board.
"""
default_str = board.__str__()
replaced_str = replace_with_unicode(default_str)
replaced_and_guided_str = guide_with_numbers(replaced_str)
properly_flipped_str = (
replaced_and_guided_str if is_white_on_bottom else replaced_and_guided_str[::-1]
)
trimmed_str = trim_whitespace_before_newline(properly_flipped_str)
monospaced_str = f"```\n{trimmed_str}\n```"
return monospaced_str
def guide_with_numbers(board_str: str) -> str:
"""Adds numbers and letters on the side of a string without them made out
of a board.
Parameters:
- board_str: The string from the board object.
Returns: The string with the numbers and letters.
"""
# Spaces and newlines would mess up the loop because they add extra indexes
# between pieces. Newlines are added later by the loop and spaces are added
# back in at the end.
board_without_whitespace_str = board_str.replace(" ", "").replace("\n", "")
# The first number, 8, needs to be added first because it comes before a
# newline. From then on, numbers are inserted at newlines.
row_list = list("8" + board_without_whitespace_str)
for i, char in enumerate(row_list):
# `(i + 1) % 10 == 0` if it is the end of a row, i.e., the 10th column
# since lists are 0-indexed.
if (i + 1) % 10 == 0:
# Since `i + 1` is always a multiple of 10 (because index 0, 10,
# 20, etc. is the other row letter and 1-8, 11-18, 21-28, etc. are
# the squares), `(i + 1) // 10` is the inverted row number (1 when
# it should be 8, 2 when it should be 7, etc.), so therefore
# `9 - (i + 1) // 10` is the actual row number.
row_num = 9 - (i + 1) // 10
# The 3 separate components are split into only 2 elements so that
# the newline isn't counted by the loop. If they were split into 3,
# or combined into just 1 string, the counter would become off
# because it would be counting what is really 2 rows as 3 or 1.
row_list[i:i] = [str(row_num) + "\n", str(row_num - 1)]
# 1 is appended to the end because it isn't created in the loop, and lines
# that begin with spaces have their spaces removed for aesthetics.
row_str = (" ".join(row_list) + " 1").replace("\n ", "\n")
# a, b, c, d, e, f, g, and h are easy to add in.
row_and_col_str = " a b c d e f g h \n" + row_str + "\n a b c d e f g h "
return row_and_col_str
def replace_with_unicode(board_str: str) -> str:
"""Replaces the default characters in a board object's string output with
Unicode chess characters, e.g., '' instead of 'R.'
Parameters:
- board_str: The string from the board object.
Returns: The string with the replaced characters.
"""
replaced_str = board_str
replaced_str = replaced_str.replace("P", "")
replaced_str = replaced_str.replace("N", "")
replaced_str = replaced_str.replace("B", "")
replaced_str = replaced_str.replace("R", "")
replaced_str = replaced_str.replace("Q", "")
replaced_str = replaced_str.replace("K", "")
replaced_str = replaced_str.replace("p", "")
replaced_str = replaced_str.replace("n", "")
replaced_str = replaced_str.replace("b", "")
replaced_str = replaced_str.replace("r", "")
replaced_str = replaced_str.replace("q", "")
replaced_str = replaced_str.replace("k", "")
replaced_str = replaced_str.replace(".", "·")
return replaced_str
def trim_whitespace_before_newline(str_to_trim: str) -> str:
"""Removes any spaces before a newline in a string.
Parameters:
- str_to_trim: The string to trim.
Returns: The trimmed string.
"""
return re.sub(r"\s+$", "", str_to_trim, flags=re.M)