
from tasks.checks.sweep import (
    check_sweep_success,
    check_relocate_success,
)
from exceptions import SubtaskSkip

from time import sleep

def run_task(env, task, provider=None, subtask_results=None, max_repairs=5, **kwargs):
    if subtask_results is None:
        subtask_results = []

    print(f"[Sweep] Starting task with pieces: {task.object_names}")
    print(f"[Sweep] Piece to box mapping: {task.piece_to_box}")
    print(f"[Sweep] Misplaced pieces: {task.obst_pieces}")

    # Step 1: Relocate misplaced pieces back to the board
    if task.obst_pieces:
        print(f"\n[Sweep] Relocating {len(task.obst_pieces)} misplaced piece(s)")

        # Calculate grid positions to avoid collisions
        board_bbox = env.get_obj_bbox("board")
        if board_bbox is not None:
            center = (board_bbox[0] + board_bbox[1]) / 2
            bbox_size = board_bbox[1] - board_bbox[0]
            spacing = min(bbox_size[0], bbox_size[1]) * 0.3
            height = board_bbox[1][2] + 0.08

            grid_offsets = [
                (0, 0), (1, 0), (-1, 0), (0, 1), (0, -1),
                (1, 1), (-1, 1), (1, -1), (-1, -1),
            ]

        for i, piece_name in enumerate(task.obst_pieces):
            print(f"[Sweep] Relocating {piece_name} back to board")

            # Calculate target position with offset
            target_pos = None
            if board_bbox is not None and i < len(grid_offsets):
                offset = grid_offsets[i]
                target_pos = center.copy()
                target_pos[0] += offset[0] * spacing
                target_pos[1] += offset[1] * spacing
                target_pos[2] = height
                print(f"[Sweep] Target position for {piece_name}: offset {offset}")

            sleep(10)
            try:
                result = provider.execute_subtask(
                    subtask_name="relocate_single",
                    env=env,
                    obj_name=piece_name,
                    target_name="board",
                    target_position=target_pos,
                    check_func=lambda e, o, t: check_relocate_success(e, o, t),
                    max_repairs=max_repairs,
                )
                subtask_results.append(result)
            except SubtaskSkip as e:
                print(f"[Sweep] Skipping relocate for {piece_name}")
                if e.result is not None:
                    subtask_results.append(e.result)
                continue

    # Step 2: Sweep all pieces to their corresponding boxes
    for piece_name in task.object_names:
        print(f"\n[Sweep] Processing: {piece_name}")

        # Get target box from piece_to_box mapping
        target_box = task.piece_to_box.get(piece_name)
        if target_box is None:
            print(f"[Sweep] No target box for {piece_name}, skipping")
            continue

        sleep(10)
        # Sweep piece to target box
        try:
            result = provider.execute_subtask(
                subtask_name="sweep_single",
                env=env,
                obj_name=piece_name,
                target_name=target_box,
                check_func=lambda e, o, t: check_sweep_success(e, o, t),
                max_repairs=max_repairs,
            )
            subtask_results.append(result)
        except SubtaskSkip as e:
            print(f"[Sweep] Skipping sweep for {piece_name}")
            if e.result is not None:
                subtask_results.append(e.result)
            continue

    # Check result
    result = env.check_result()
    print(f"\n[Sweep] Task completed with result: {result}")
    return result
