How to make a Lichess bot in Python

How to make a Lichess bot in Python

Chess is one of the most popular strategic games played all around the world. Chess bots now dominate even the highest level grand-masters so how do we make one?

To get started all you’ll need is Python3.9 and Github installed!

Outline

Lichess Communication:

Setting up a communication to lichess using lichess-bot.

Chess Bot:

Building a minimax chess engine using position and material eval.

Hosting:

Using railway.app to host your chess bot.

Lichess Communication

Cloning

To start off we’re going to set up a brand new VSCode project!

We’re going to use lichess-bot to set up a live data stream to lichess so that we can play games withing their application.

Clone github.com/lichess-bot-devs/lichess-bot.git into your new project! Make sure you have github installed!

git clone https://github.com/lichess-bot-devs/lichess-bot.git

Virtual Env

Now we’re going to set up a consistent virtual env for python.

This will help with maintaining consistency and managing dependencies in our application.

cd lichess-bot
pip install virtualenv
python -m venv venv
venv\Scripts\activate
pip install -r requirements.txt

Once this is done, copy config.yml.default to config.yml

Lichess Bot Account

Once you’re finished, we’re going to set up a lichess bot account. Head over to https://lichess.org/ and create a new account.

It’s important that this account has not games played in order to become a bot account.

Once you’ve create an account, go to preferences -> API access tokens and create a key that looks like this.

Now put the following curl command into terminal and replace YOUR_TOKEN with the token you get after creating.

Make sure to save this token into a save place too, because we’re going to be using it again!

curl -d '' https://lichess.org/api/bot/account/upgrade -H "Authorization: Bearer YOUR_TOKEN"

Head back to lichess and you should now have a bot account!!

Now that you have a bot account, add your key we created earlier (you can create a new one with bot scope too) into the lichess config.yml file. Replace xxxxxxxxxx with your key:

Playing against it

Now that you have your bot all set up, it’s time to try it out.

In config.yml, switch name to “RandomMove” and protocol to “homemade”.

We want to be able to play most games against our bot so uncomment correspondance and switch 14 days to .inf.

Then run inside of your terminal:

python3 lichess-bot.py

You’re now connected to lichess and ready to start playing some games against your bot!!

Note: You’ll need to switch your lichess account to play against your bot.

But right now you’re bot plays random moves, let’s make it STRONG now.

Building the AI

How it’s going to work

To build the AI, we’re going to be using a minimax search with position and material eval, pretty simple but strong.

Minimax will basically search through millions of possible positions from the current board, evaluate them and then give us which one is best. Think of it as a massive tree.

Evaluation is how we’re going to determine whether this position of millions of others is better than the others.

Position eval takes every piece in that position, finds what square it’s on and then gives it a point rating.

Each of these numbers are a spot on the board and how much the piece is worth on that square.

Then we add this value to their material eval:

You’ll notice I usually give the bishop 330, just to differentiate it from the knight and because they’re usually stronger later into the endgame.

And this is how we’re going to evaluate our positions.

Let’s get programming

Create a new folder inside of /engines called bot. Inside of this folder, create a file called main.py.

Position Evaluation

Create a new file inside of /bot called positions.py. Inside of the file, put all the positions eval for each piece:

# Pawn
pawn = [
    0, 0, 0, 0, 0, 0, 0, 0,
    5, 10, 10, -20, -20, 10, 10, 5,
    5, -5, -10, 0, 0, -10, -5, 5,
    0, 0, 0, 20, 20, 0, 0, 0,
    5, 5, 10, 25, 25, 10, 5, 5,
    10, 10, 20, 30, 30, 20, 10, 10,
    50, 50, 50, 50, 50, 50, 50, 50,
    0, 0, 0, 0, 0, 0, 0, 0]


# Knight
knight = [
    -50, -40, -30, -30, -30, -30, -40, -50,
    -40, -20, 0, 5, 5, 0, -20, -40,
    -30, 5, 10, 15, 15, 10, 5, -30,
    -30, 0, 15, 20, 20, 15, 0, -30,
    -30, 5, 15, 20, 20, 15, 5, -30,
    -30, 0, 10, 15, 15, 10, 0, -30,
    -40, -20, 0, 0, 0, 0, -20, -40,
    -50, -40, -30, -30, -30, -30, -40, -50]


# Bishop
bishop = [
    -20, -10, -10, -10, -10, -10, -10, -20,
    -10, 5, 0, 0, 0, 0, 5, -10,
    -10, 10, 10, 10, 10, 10, 10, -10,
    -10, 0, 10, 10, 10, 10, 0, -10,
    -10, 5, 5, 10, 10, 5, 5, -10,
    -10, 0, 5, 10, 10, 5, 0, -10,
    -10, 0, 0, 0, 0, 0, 0, -10,
    -20, -10, -10, -10, -10, -10, -10, -20]


# Rook
rook = [
    0, 0, 0, 5, 5, 0, 0, 0,
    -5, 0, 0, 0, 0, 0, 0, -5,
    -5, 0, 0, 0, 0, 0, 0, -5,
    -5, 0, 0, 0, 0, 0, 0, -5,
    -5, 0, 0, 0, 0, 0, 0, -5,
    -5, 0, 0, 0, 0, 0, 0, -5,
    5, 10, 10, 10, 10, 10, 10, 5,
    0, 0, 0, 0, 0, 0, 0, 0]


# Queen
queen = [
    -20, -10, -10, -5, -5, -10, -10, -20,
    -10, 0, 0, 0, 0, 0, 0, -10,
    -10, 5, 5, 5, 5, 5, 0, -10,
    0, 0, 5, 5, 5, 5, 0, -5,
    -5, 0, 5, 5, 5, 5, 0, -5,
    -10, 0, 5, 5, 5, 5, 0, -10,
    -10, 0, 0, 0, 0, 0, 0, -10,
    -20, -10, -10, -5, -5, -10, -10, -20]


# King
king = [
    20, 30, -5, 0, -5, -5, 30, 20,
    20, 20, -5, -5, -5, -5, 20, 20,
    -10, -20, -20, -20, -20, -20, -20, -10,
    -20, -30, -30, -40, -40, -30, -30, -20,
    -30, -40, -40, -50, -50, -40, -40, -30,
    -30, -40, -40, -50, -50, -40, -40, -30,
    -30, -40, -40, -50, -50, -40, -40, -30,
    -30, -40, -40, -50, -50, -40, -40, -30]

If you want your bot to play a specific way, you can alter these values but these will give you a generally good bot.

Material Evaluation

Now we’re going to give each of our pieces a value.

Create a new file called material.py inside of bot. Pieces are going to get the following values:

  • Pawn: 100

  • Knights: 310

  • Bishops: 330

  • Rook: 500

  • Queen: 900

Inside of material.py, add the following code:

import chess

def get_material(board):
    # Weights
    pw = 100
    kw = 310
    bw = 330
    rw = 500
    qw = 900
    kingw = 20000

    wp = len(board.pieces(chess.PAWN, chess.WHITE))
    wr = len(board.pieces(chess.ROOK, chess.WHITE))
    wk = len(board.pieces(chess.KNIGHT, chess.WHITE))
    wb = len(board.pieces(chess.BISHOP, chess.WHITE))
    wq = len(board.pieces(chess.QUEEN, chess.WHITE))
    wking = len(board.pieces(chess.KING, chess.WHITE))

    bp = len(board.pieces(chess.PAWN, chess.BLACK))
    br = len(board.pieces(chess.ROOK, chess.BLACK))
    bk = len(board.pieces(chess.KNIGHT, chess.BLACK))
    bb = len(board.pieces(chess.BISHOP, chess.BLACK))
    bq = len(board.pieces(chess.QUEEN, chess.BLACK))
    bking = len(board.pieces(chess.KING, chess.BLACK))

    # White
    wpw = wp * pw
    # Rook weight increases for less pawns
    wrw = wr * rw
    # Knight weight goes down for each enemy pawn gone (8 pawns)
    wkw = wk * kw
    wbw = wb * bw
    wqw = wq * qw
    wkingw = wking * kingw

    # Black
    bpw = bp * pw
    # Rook weight increases for less pawns
    brw = br * rw
    # Knight weight goes down for each enemy pawn gone  (8 pawns)
    bkw = bk * kw
    bbw = bb * bw
    bqw = bq * qw
    bkingw = bking * kingw

    white_material = wpw + wrw + wkw + wbw + wqw + wkingw
    black_material = bpw + brw + bkw + bbw + bqw + bkingw

    total_material = white_material - black_material

    return total_material

First, we assign our weights to our pieces. Next we grab all of those pieces on the board. Finally we do we calculations for the total piece eval of each side and make the material eval.

The final eval function

Now using the positions and the material eval, we’re going to calulcate the board evaluation.

Create a new file called evaluation.py. Put the following code inside.

from .material import get_material
import chess
from . import positions

def get_evaluation(board):

    # Check for checkmate of the opponent
    if board.is_checkmate():
        if board.turn:
            return -9999
        else:
            return 9999
    if board.is_stalemate():
            return 0
    if board.is_insufficient_material():
            return 0

    total_material = get_material(board)

    pawnsq = sum([positions.pawn[i] for i in board.pieces(chess.PAWN, chess.WHITE)])
    pawnsq = pawnsq + sum([-positions.pawn[chess.square_mirror(i)]
                        for i in board.pieces(chess.PAWN, chess.BLACK)])
    knightsq = sum([positions.knight[i] for i in board.pieces(chess.KNIGHT, chess.WHITE)])
    knightsq = knightsq + sum([-positions.knight[chess.square_mirror(i)]
                            for i in board.pieces(chess.KNIGHT, chess.BLACK)])
    bishopsq = sum([positions.bishop[i] for i in board.pieces(chess.BISHOP, chess.WHITE)])
    bishopsq = bishopsq + sum([-positions.bishop[chess.square_mirror(i)]
                            for i in board.pieces(chess.BISHOP, chess.BLACK)])
    rooksq = sum([positions.rook[i] for i in board.pieces(chess.ROOK, chess.WHITE)])
    rooksq = rooksq + sum([-positions.rook[chess.square_mirror(i)]
                        for i in board.pieces(chess.ROOK, chess.BLACK)])
    queensq = sum([positions.queen[i] for i in board.pieces(chess.QUEEN, chess.WHITE)])
    queensq = queensq + sum([-positions.queen[chess.square_mirror(i)]
                            for i in board.pieces(chess.QUEEN, chess.BLACK)])
    kingsq = sum([positions.king[i] for i in board.pieces(chess.KING, chess.WHITE)])
    kingsq = kingsq + sum([-positions.king[chess.square_mirror(i)]
                        for i in board.pieces(chess.KING, chess.BLACK)])

    eval = total_material + pawnsq + knightsq + rooksq + queensq + kingsq 

    return eval

Let’s break this function down:

  1. If the board is a stalemate return 0, if it’s checkmate, return an impossibly high eval.

  2. If it’s neither of these scenarios calculate the position evals and add the material eval to it.

  3. Return the eval to be used

Openings

Bots sometimes struggle with variety and openings so I like to give it a bunch of openings to play. This isn’t required but I prefer to add it in.

First download this dataset of 2700 openings: https://drive.google.com/file/d/1zDjc9O3QmqvVKfBZ1fQsjXJG6ThwUAYa/view?usp=sharing

Next, add it into your workspace in the /bot folder. Your workspace should look like this at this point:

Create a new file inside /bot now called opening.py. Add the following code to it:

import pandas as pd
import chess
import chess.pgn
import random
import os


def play_opening(board):
    next_opening_moves = [];

    # If we go first, we just play e4
    if board.turn == chess.WHITE and board.fullmove_number == 1:
        next_opening_moves.append("e2e4")

    new_board = chess.Board()

    # Get the current directory of game.py
    current_directory = os.path.dirname(os.path.abspath(__file__))

    # Define the file path relative to the current directory
    file_path = os.path.join(current_directory, 'openings.csv')

    # Get all of the SAN notations
    chess_openings = pd.read_csv(file_path)

    chess_openings = chess_openings["moves"].tolist()

    # Loop over each opening
    # If it "contains" the same board position as our current board
    # Return it's next move
    for opening in chess_openings:
        moves_in_openings = opening.split();

        for index, move in enumerate(moves_in_openings):
            try:
                new_board.push_san(move)

                if board == new_board:
                    next_move = board.parse_san(moves_in_openings[index + 1]).uci()
                    next_opening_moves.append(next_move)
            except:
                break;


        new_board.reset()


    # If there are no more opening moves, return None
    if not next_opening_moves:
        return None

    # If there is valid openings, randomly choose the next move of them
    random_opening_from_array = random.choice(next_opening_moves)

    return random_opening_from_array

First, if we’re white, it will play e4 (My fav move yk). Next it will read the CSV file and if the opening is inside of here, it will play the next move in the sequence. If there’s no more opening moves, then it will allow our evaluation function to continue working.

Minimax function

Our minimax function is going to let us find the next best move in the future based off millions of possible moves.

Create a new file called minimax.py and put the following code inside:

from .eval import get_evaluation
import numpy as np

def minimax(board, depth, alpha, beta, maximizing_player):
  if depth == 0 or board.is_game_over():
    return get_evaluation(board)

  if maximizing_player:
    max_eval = -np.inf
    for move in board.legal_moves:
      board.push(move)
      eval = minimax(board, depth - 1, alpha, beta, False)
      board.pop()
      max_eval = max(max_eval, eval)
      alpha = max(alpha, eval)
      if beta <= alpha:
        break
    return max_eval
  else:
    min_eval = np.inf
    for move in board.legal_moves:
      board.push(move)
      eval = minimax(board, depth - 1, alpha, beta, True)
      board.pop()
      min_eval = min(min_eval, eval)
      beta = min(beta, eval)
      if beta <= alpha:
        break
    return min_eval

Basically this function will take in a board depth (how many moves to look in the future) and then will find every single possible move based off of this.

It will also use alpha beta pruning to speed it up and optimise it for only the top tier moves.

If you want to understand more about this, check out the chess programming wiki: chessprogramming.org/Minimax

Let’s get it playing!

Navigate to /main.py and add the following code:

import chess
import numpy as np

from .opening import play_opening
from .minimax import minimax


def get_move(board, depth):
    opening_move = play_opening(board)

    if opening_move:
        print("PLAYING OPENING MOVE: ", opening_move)
        return opening_move

    top_move = None;

    # Opposite of our minimax
    if board.turn == chess.WHITE:
      top_eval = -np.inf
    else:
      top_eval = np.inf

    for move in board.legal_moves:
        board.push(move)

        # WHEN WE ARE BLACK, WE WANT TRUE AND TO GRAB THE SMALLEST VALUE
        eval = minimax(board, depth - 1, -np.inf, np.inf, board.turn)

        board.pop()

        if board.turn == chess.WHITE:
            if eval > top_eval:
                top_move = move
                top_eval = eval
        else:
            if eval < top_eval:
                top_move = move
                top_eval = eval

    print("CHOSEN MOVE: ", top_move, "WITH EVAL: ", top_eval)
    return top_move

First, if there’s an opening, we’re going to play it. If not, we’re going to use our minimax function to find the best move.

Next, navigate to /lichess-bot/homemade.py. We’re going to add our engine into this so we can use it with the lichess api.

At the top of this file, import your engine in:

from engines.bot.main import get_move

Now add your engine strategy into the file, make sure you put this BELOW the ExampleEngine class:

class PyBot(ExampleEngine):
    def search(self, board: chess.Board, time_limit: Limit, ponder: bool, draw_offered: bool, root_moves: MOVE) -> PlayResult:
        print("GETTING MOVE!")

        move = get_move(board, 4)

        return PlayResult(move, None)

Finally, switch the engine name inside of config.yml to PyBot.

You can now play your bot!! Run the server with (make sure you’re in the directory):

python3 lichess-bot.py

Now go to lichess.orgyour-bot and challenge it to a game!

Hosting

Now if you want to take this a step farther, you can host it so that you never have to have the server run yourself.

Docker

We’re going to be using a custom dockerfile in order to host everything. This will ensure consistency from our dev env to the providers env.

Inside of your root directory, create a file called Dockerfile (with uppercase D) and add the following instructions:

# Use an official Python runtime as a parent image
FROM python:3.9-slim

# Set environment variables
ENV PYTHONDONTWRITEBYTECODE 1
ENV PYTHONUNBUFFERED 1
ENV NIXPACKS_PATH /opt/venv/bin:$NIXPACKS_PATH

# Set the working directory in the container
WORKDIR /app

# Install system dependencies
RUN apt-get update \
    && apt-get install -y --no-install-recommends \
    build-essential \
    && apt-get clean \
    && rm -rf /var/lib/apt/lists/*

# Copy the current directory contents into the container at /app
COPY . /app/

# Create a virtual environment and set it as the Python environment
RUN python -m venv /opt/venv
ENV PATH="/opt/venv/bin:$PATH"

# Install Python dependencies
RUN pip install --upgrade pip \
    && pip install -r requirements.txt

# Command to run the application
CMD ["python", "lichess-bot.py"]

You’re project should now be looking like this:

Github

We need to push our entire project to github so our hosting can connect to it and run it.

Navigate to github.com and create a new repository:

Next go back to your code and tap the .gitignore. Remove the following lines:

*.yml
/engines/*

Now let’s update the requirements.txt to make sure they get added when we build the Dockerfile. Add these to it:

numpy==1.26.0
pandas==2.2.2

It should look something like this now:

Then run the following commands:

git reset -- lichess-bot
git add .
git commit -m "hosting"
git remote add origin https://github.com/username/repository.git - CHANGE THIS
git push --set-upstream origin master

This will initialize the repo, remove the submodule, add the files and then push them to Github.

Let’s get hosting

Next let’s get hosting! Go to railway.app and create an account. Now create a new project and select your repo:

Now check the build logs and make sure everything is good and you’re chess bot should be done!

Congratulation on making you’re first chess bot!

Next steps

Now if you want to improve your chess bot, you can head on over to the chess programming wiki chessprogramming.org/Main_Page and check out all the ways to improve the bot.

You can also go back through the code and make it more efficient so it’s faster and can have a farther depth.

Conclusion

If you enjoyed this detailed tutorial and interested in more of my stuff, you can check out my website: kaipereira.com or my Github github.com/KaiPereira

Did you find this article valuable?

Support Kai Pereira by becoming a sponsor. Any amount is appreciated!