import numpy as np


def str_to_board(sudoku_str):
    """Convert Kaggle string to 9x9 array ('.' converted to 0)"""
    if len(sudoku_str) != 81:
        raise ValueError(f"String length must be 81, got {len(sudoku_str)}")
    board = []
    for i in range(9):
        row = [0 if c == '.' else int(c) for c in sudoku_str[i*9 : (i+1)*9]]
        board.append(row)
    return np.array(board, dtype=int)


def solve_sudoku(input_problem):
    """
    Solve Sudoku puzzle (ensure returned solution has no 0, return empty list if unsolvable)
    """
    if isinstance(input_problem, str):
        try:
            board = str_to_board(input_problem)
            input_type = list
        except ValueError as e:
            print(f"String conversion failed: {e}")
            return []
    elif isinstance(input_problem, list):
        board = np.array(input_problem, dtype=int)
        input_type = list
    elif isinstance(input_problem, np.ndarray):
        board = input_problem.copy()
        input_type = np.ndarray
    else:
        raise TypeError("Input must be str, list or np.ndarray")

    if not is_valid_board(board):
        return []

    if backtrack_solve(board):
        if np.count_nonzero(board == 0) == 0:
            return board.tolist() if input_type == list else board
        else:
            print("Warning: Solver returned incomplete solution (contains 0), treated as unsolvable")
            return []
    return []


def is_valid_board(board):
    """Check the validity of the initial Sudoku board"""
    for i in range(9):
        for j in range(9):
            val = board[i][j]
            if val != 0:
                original_val = val
                board[i][j] = 0
                if not is_valid_move(board, original_val, (i, j)):
                    board[i][j] = original_val
                    return False
                board[i][j] = original_val
    return True


def is_valid_move(board, num, pos):
    """Check if a number can be placed at the specified position"""
    row, col = pos
    if num in board[row]:
        return False
    if num in board[:, col]:
        return False
    box_row, box_col = (row // 3) * 3, (col // 3) * 3
    if num in board[box_row:box_row+3, box_col:box_col+3]:
        return False
    return True


def get_candidates(board, pos):
    """Get candidate numbers for the specified position"""
    row, col = pos
    candidates = []
    for num in range(1, 10):
        if is_valid_move(board, num, (row, col)):
            candidates.append(num)
    return candidates


def find_best_empty_cell(board):
    """Select the empty cell with the fewest candidate numbers"""
    min_candidates = 10
    best_pos = None
    for i in range(9):
        for j in range(9):
            if board[i][j] == 0:
                candidates = get_candidates(board, (i, j))
                if len(candidates) < min_candidates:
                    min_candidates = len(candidates)
                    best_pos = (i, j)
                    if min_candidates == 1:
                        return best_pos
    return best_pos


def backtrack_solve(board):
    """Backtracking core for Sudoku solving"""
    empty_pos = find_best_empty_cell(board)
    if not empty_pos:
        return np.count_nonzero(board == 0) == 0

    row, col = empty_pos
    for num in get_candidates(board, (row, col)):
        board[row][col] = num
        if backtrack_solve(board):
            return True
        board[row][col] = 0
    return False


# ------------------------------
# Minimal Usage Example 
# ------------------------------
# if __name__ == "__main__":
#     # Example: Initialize and solve a Sudoku puzzle
#     puzzle_str = ".5..83.17...1..4..3.4..56.8....3...9.9.8245....6....7...9....5...729..861.36.72.."
#     board = Board(puzzle_str)
#     if board.solve():
#         print("Solved Sudoku board:")
#         print(board.board)
#     else:
#         print("Failed to solve the puzzle")