Ferramentas de análise e revisão de partidas de xadrez baseadas em Stockfish.
Project description
Saulochess - Chess Analysis Engine
saulochess is a robust Python API for detailed chess game analysis and review, utilizing the powerful Stockfish engine to classify moves and generate analysis comments (such as brilliant, mistake, blunder, etc.).
The package is designed to be flexible and free from proprietary data dependencies. The end-user is responsible for providing the Stockfish path and their own open-source openings database to enable the openings functionality.
Installation
The package is available on PyPI.
pip install saulochess
Prerequisites
Stockfish Engine: You must have the Stockfish executable (version 16 or higher is recommended) installed and accessible. You will need to provide the path to this executable when initializing the engine.
API Usage
The main module is chess_review. We recommend that you always start and close the Stockfish engine manually to manage resources efficiently.
- Single Move Review (review_move) This function analyzes the current board position and classifies a specific move, returning Stockfish's best move for comparison.
import chess.engine
from saulochess import chess_review
import chess
# --- Configuration ---
STOCKFISH_PATH = r"C:\path\to\stockfish.exe" # Update this path
try:
# 1. Initialize the Stockfish Engine with 'with'
with chess.engine.SimpleEngine.popen_uci(STOCKFISH_PATH) as engine:
# 2. Set up the Board and the Move
board = chess.Board("rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1") # Initial position
move = chess.Move.from_uci("e2e4") # The move to be analyzed
# 3. Review the Move
classification, review, best_move, san_best = chess_review.review_move(
board,
move,
previous_review="Game Start",
engine=engine # **REQUIRED: Pass the open engine object**
language='en' # NEW - Choose your language: 'en' or 'ptbr'
)
# 4. Result
print(f"Classification: {classification}")
print(f"Engine Review: {review}")
print(f"Stockfish Best Move: {san_best} ({best_move})")
# 'with' automatically calls engine.quit() here.
except Exception as e:
print(f"ERROR: Could not open Stockfish or analyze. Check the path. Details: {e}")
- Full Game Analysis (pgn_game_review) The main game analysis function takes a game in PGN string format and returns a tuple containing all analysis data, including accuracy, ELO estimation, and the full list of classifications and comments (18 elements total).
import chess.engine
from saulochess import chess_review # Assumes the library is installed
import platform
import traceback
# ---------------------------------------------------------------------------
# IMPORTANT: REQUIRED CONFIGURATION
# ---------------------------------------------------------------------------
# 1. Change this variable to the FULL path of the Stockfish executable
STOCKFISH_PATH = r'path\\to\\stockfish.exe' # UPDATE THIS PATH
# Analysis Parameters
ANALYSIS_LIMIT_TYPE = 'depth' # Or 'Time'. We strongly recommend you to use depth, because it's faster and does not impact in the review.
ANALYSIS_TIME_LIMIT = 0.2 # Time in seconds per move (used if ANALYSIS_LIMIT_TYPE="time")
ANALYSIS_DEPTH_LIMIT = 10 # Depth (used if ANALYSIS_LIMIT_TYPE="depth")
# Example PGN for analysis (The Immortal Game)
PGN_EXAMPLE = """
[Event "London, 1851"]
[Site "London ENG"]
[Date "1851.06.21"]
[Round "6"]
[White "Adolf Anderssen"]
[Black "Lionel Adalbert Bagelsen"]
[Result "1-0"]
1. e4 e5 2. f4 exf4 3. Bc4 Qh4+ 4. Kf1 b5 5. Bxb5 Nf6 6. Nf3 Qh6 7. d3 Nh5 8. Nh4 Qg5 9. Nf5 c6 10. g4 Nf6 11. Rg1 cxb5 12. h4 Qg6 13. h5 Qg5 14. Qf3 Nc6 15. Bxf4 Ne5 16. Bxe5 Qc1+ 17. Kf2 Qxc2+ 18. Ke1 Bb4+ 19. Nc3 Bxc3+ 20. bxc3 d6 21. Bxf6 gxf6 22. e5 fxe5 23. Qxa8 O-O 24. Ne7+ Kh8 25. Nxc8 Qxc3+ 26. Ke2 Qc2+ 27. Kf3 Qxd3+ 28. Kf2 Qd2+ 29. Kf1 Qf4+ 30. Ke2 e4 31. Raf1 Qh2+ 32. Ke3 Qe5 33. Rf5 Qc3+ 34. Kxe4 Qc2+ 35. Kd4 Qb2+ 36. Kd5 Qa3 37. Rgf1 h6 38. Rxf7 Qd3+ 39. Kc6 Qc4+ 40. Kb7 Rf7+ 41. Rxf7 Qxf7+ 42. Kxa7 Kg7 43. Ka6 Qxa2+ 44. Kb6 b4 45. Nxd6 b3 46. Nf5+ Kf6 47. Nxh6+ Kg5 48. Nf7+ Kxg4 49. Ne5+ Kh3 50. Nc4 Qc2 51. Na3 Qc1 52. Nb5 b2 53. Nd4 b1=Q 54. Nxb1 Qxb1 55. Qh8+ Kg2 56. Qg7+ Kf1 57. Qf6+ Ke2 58. Qe5+ Kd1 59. Qd4+ Kc1 60. Qc4+ Kb2 61. Qb4+ Ka1 62. Qa3+ Qa2 63. Qxa2+ Kxa2 64. h6 Kb3 65. h7 Kc4 65. h8=Q Kb4 67. Qh4+ Ka3 68. Qb8 Ka2 69. Kc5 Ka3 70. Kc4 Ka2 71. Kc3 Ka1 72. Kc2 Ka2 73. Qb3+ Ka1 74. Qb2# 1-0
"""
# ---------------------------------------------------------------------------
def run_test():
print("Starting chess_review API test...")
# A simple check to see if the user has updated the path
if "path\\to\\stockfish.exe" in STOCKFISH_PATH:
print("\n--- CONFIGURATION ERROR ---")
print("Please edit the 'STOCKFISH_PATH' variable with the correct path.")
return
try:
print(f"Opening engine at: {STOCKFISH_PATH}...")
# Initialize the Stockfish Engine using 'with' to ensure safe closure
with chess.engine.SimpleEngine.popen_uci(STOCKFISH_PATH) as engine:
print(f"Engine opened successfully. Version: {engine.id['name']}")
print("\nStarting pgn_game_review...")
# 3. Analyze the Full Game, PASSING ALL 6 ARGUMENTS
game_data = chess_review.pgn_game_review(
pgn_data=PGN_EXAMPLE,
roast=False,
limit_type=ANALYSIS_LIMIT_TYPE,
time_limit=ANALYSIS_TIME_LIMIT,
depth_limit=ANALYSIS_DEPTH_LIMIT,
engine=engine # The 'engine' object is passed to the API
language='en' # NEW - Choose your language: 'en' or 'ptbr'
)
print("Function pgn_game_review completed.")
# 4. Unpack the 18 returned values
(
san_moves, fens, scores, classification_list, review_list, best_review_list,
san_best_moves, uci_best_moves, devs, tens, mobs, conts,
white_acc, black_acc, white_elo_est, black_elo_est,
average_cpl_white, average_cpl_black
) = game_data
print("\n--- GENERAL RESULTS ---")
print(f"Game analyzed with {len(san_moves)} moves.")
print(f"White Accuracy: {white_acc:.2f}% (Est. ELO: {white_elo_est})")
print(f"Black Accuracy: {black_acc:.2f}% (Est. ELO: {black_elo_est})")
print(f"Average CPL White: {average_cpl_white:.2f}")
print(f"Average CPL Black: {average_cpl_black:.2f}")
print("\n--- MOVE-BY-MOVE REVIEW ---")
if not san_moves:
print("No moves were analyzed (empty lists returned).")
return
# Display results by combining lists
for i, san in enumerate(san_moves):
try:
classif = classification_list[i].upper() if i < len(classification_list) else "N/A"
review = review_list[i] if i < len(review_list) else "(N/A)"
best_move = san_best_moves[i] if i < len(san_best_moves) else "N/A"
print(f" {i+1}. {san} ({classif}): {review}")
print(f" Best Move: {best_move} | Score: {scores[i]}")
except IndexError:
# Catches cases where one list is shorter than another (internal API error)
print(f" {i+1}. {san}: (INCOMPLETE ANALYSIS DATA)")
except Exception as e_loop:
print(f" {i+1}. {san}: (Error printing data: {e_loop})")
except FileNotFoundError:
print("\n--- CRITICAL ERROR ---")
print(f"The Stockfish engine was not found at path: {STOCKFISH_PATH}")
except Exception as e:
print(f"\n--- UNEXPECTED ERROR DURING ANALYSIS ---")
print(f"An error occurred: {e}")
traceback.print_exc()
print("\nTest finished.")
if __name__ == "__main__":
try:
import chess
except ImportError:
print("Error: The 'python-chess' library is not installed.")
print("Please install it using: pip install python-chess")
else:
run_test()
- Openings Integration (Optional Feature) To use the openings identification feature, you must load your own Pandas DataFrame and pass it as an argument to the review functions:
import pandas as pd
from saulochess import chess_review
# ... (Engine initialization)
# 1. Load your own opening database.
# Your CSV must have 'pgn' and 'name' columns.
try:
my_openings_df = pd.read_csv("path/to/my_openings_database.csv")
except:
my_openings_df = None
engine = chess.engine.open_uci("stockfish")
# 2. Use the feature by passing the DataFrame:
classification, review, best_move, san_best = chess_review.review_move(
board,
move,
previous_review="Initial",
engine=engine,
openings_df=my_openings_df, # <--- PASS THE DATAFRAME HERE
check_if_opening=True # <--- ACTIVATE THE CHECK HERE
language='en'
)
engine.quit()
⚠️ Known Bug: First Move Analysis
We are currently aware of a minor bug where the analysis of the first move of the game may fail internally, often resulting in an argument of type 'NoneType' is not iterable warning/error.
Symptom: The console may show an internal [AVISO] message or an ERROR classification for the first move only.
Impact: The rest of the game analysis (move 2 onwards) is generally unaffected, and the final summary results (Accuracy, ELO, Avg CPL) are correctly calculated based on the successful analysis of the remaining moves.
We are working on a fix to ensure a smooth analysis start.
License
This package uses base code licensed under the MIT License. The copyright notice for the original base code is included in the LICENSE file: Copyright (c) 2023 LinkAnJarad.
Further developments are Copyright (c) 2025 Saulo/SauloChess and are also under the MIT License.
The original code is from LinkAnJarad, available in OpenChess-Insights. I made some alterations, such as improvements of the engine management and translated it to portuguese.
By the way, the API code is open-source. If you want to contribute, go to https://github.com/Git49-max/saulochess. Don't forget to see the OpenChess-Insights github too!
Buy LinkAnJarad a coffee: https://buymeacoffee.com/linkanjarad
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
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file saulochess-0.1.7.1.tar.gz.
File metadata
- Download URL: saulochess-0.1.7.1.tar.gz
- Upload date:
- Size: 26.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
9092223d0d2ebca160328a1985a07373af00d5971d8018cb370261a1f548d90b
|
|
| MD5 |
d0c00459485c08bdf2c55349b6cfa9da
|
|
| BLAKE2b-256 |
6db8ff65cc699bbd1aa820324be9a5abdd034465b02aadf0981370a6b51bd02e
|
File details
Details for the file saulochess-0.1.7.1-py3-none-any.whl.
File metadata
- Download URL: saulochess-0.1.7.1-py3-none-any.whl
- Upload date:
- Size: 22.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
52bd83f8aa1a7220c3408849c8d467fe6f6c358f8f113362cae268445974e1e8
|
|
| MD5 |
bd50b6dd5ab37cbe6736ab00623b40bf
|
|
| BLAKE2b-256 |
f6c1013d4dc96979bb67f887bdaca254b724cd04ea20baaa2baf9aa43676b49c
|