import chess
import chess.engine
import concurrent.futures
from typing import List, Tuple
import multiprocessing

# Global variable to store engine instance per process
engine_instance = None


def initialize_engine(eval_engine_path: str):
    """
    Initializes the chess engine for each worker process.

    :param eval_engine_path: Path to the chess engine executable.
    :return: A reference to the initialized engine.
    """
    global engine_instance
    # Initialize the engine only once per process
    if engine_instance is None:
        engine_instance = chess.engine.SimpleEngine.popen_uci(eval_engine_path)
    return engine_instance


def evaluate_board(index: int, board: chess.Board, depth: int = 10) -> Tuple[int, chess.Board, chess.Move]:
    """
    Evaluates a board position using the existing engine instance.

    :param index: The index of the board position in the original input list.
    :param board: The chess board to search.
    :param depth: The depth to which the engine should search.
    :return: A tuple containing the board and the suggested move in UCI format.
    """
    global engine_instance
    try:
        # Run the engine analysis with the specified depth
        result = engine_instance.play(board, chess.engine.Limit(depth=depth))

        # Return the suggested move in UCI format
        return index, board, result.move
    except Exception as e:
        # Return an error message if something goes wrong
        print(e)
        return index, board, None


def parallel_evaluation(boards: List[chess.Board], engine_path: str, max_workers: int = 30, depth: int = 10) -> List[
    Tuple[chess.Board, chess.Move]]:
    """
    Parallelizes the evaluation of multiple board positions, keeping a single engine instance per worker.

    :param boards: A list of FEN strings representing the boards.
    :param engine_path: Path to the chess engine executable.
    :param max_workers: Maximum number of parallel workers (typically the number of CPU cores).
    :param depth: The depth to which the engine should search.
    :return: A list of tuples with each FEN and its corresponding suggested move.
    """

    results = [None] * len(boards)

    # Define the initializer to set up the engine instance per process
    def initializer():
        initialize_engine(engine_path)

    # Use ProcessPoolExecutor to manage multiple processes with a single engine per worker
    with concurrent.futures.ProcessPoolExecutor(max_workers=max_workers, initializer=initializer) as executor:
        # Map the evaluation function to the FEN strings
        # Each task will run evaluate_board using the existing engine instance
        futures = {executor.submit(evaluate_board, idx, board, depth): idx for idx, board in enumerate(boards)}

        # Collect results as they complete
        for future in concurrent.futures.as_completed(futures):
            index, board, move_ = future.result()
            results[index] = (fen, move_)

    return results


# Example usage
if __name__ == "__main__":
    # Replace 'your_engine_path' with the actual path to your chess engine, e.g., 'stockfish'
    engine_path = 'path/to/your/engine'  # Update with your engine's path
    # Sample FEN strings (these should be your input board positions)
    sample_fens = ["r1bqkbnr/pppppppp/n7/8/8/5N2/PPPPPPPP/RNBQKB1R w KQkq - 0 2",
                   ...]  # Add your actual FEN strings here

    # Run parallel evaluation on the sample FENs
    moves = parallel_evaluation(sample_fens, engine_path, max_workers=30, depth=10)

    # Print the results
    for fen, move in moves:
        print(f"Board: {fen}, Suggested move: {move}")
